StreamAPI
概述
- Stream关注的是对数据的运算,与CPU打交道;集合关注的是数据的存储,与内存打交道
- Java 8提供了一套API,使用这套API可以对内存中的数据进行过滤、排序、映射、归约等操作。类似于sql对数据库中表的相关操作
- Stream是数据渠道,用于操作数据源(集合、数组等)所生成的元素序列。“集合讲的是数据, Stream讲的是计算”
特点
- Stream 自己不会存储元素
- Stream 不会改变源对象。相反他们会返回一个持有结果的新Stream
- Stream 操作是延迟执行的。这意味着他们会等到需要结果的时候才执行
使用流程
- 创建Stream对象
- 一系列的中间操作(过滤、映射、…)
- 终止操作
说明
- 一个中间操作链,对数据源的数据进行处理
一旦执行终止操作,就执行中间操作链,并产生结果。之后,不会再被使用
使用方法
1️⃣步骤一 创建Stream
创建方式一:通过集合
Java 8的Collection接口被扩展,提供了两个获取流的方法:**default Stream<E> stream()**
返回一个顺序流**default Stream<E> parallelStream()**
返回一个并行流
创建方式二:通过数组
Java 8中的Arrays的静态方法stream()可以获取数组流**static<T> Stream<T> stream(T[] array**)
返回一个流
重载形式,能够处理对应基本类型的数组**public static IntStream stream(int[] array)**
**public static LongStream stream(long[] array)**
**public static DoubleStream stream(double[] array)**
创建方式三:通过Stream的of()方法
可以调用Stream类静态方法of(),通过显示值创建一个流。可以用于接收任意数量的参数**public static <T> Stream<T> of(T... values)**
返回一个流
创建方式四:创建无限流迭代
**public static<T> Stream<T> iterate(final T seed, final UnaryOperator<T> f)**
- 生成
**public static<T> Stream<T> generate(Supplier<T> s)**
public class StreamAPITest {
// 创建 Stream方式一:通过集合
@Test
public void test1() {
List<Employee> employees = EmployeeData.getEmployees();
// default Stream<E> stream() : 返回一个顺序流
Stream<Employee> stream = employees.stream();
// default Stream<E> parallelStream() : 返回一个并行流
Stream<Employee> parallelStream = employees.parallelStream();
}
// 创建 Stream方式二:通过数组
@Test
public void test2() {
int[] arr = new int[]{1,2,3,4,5,6};
// 调用Arrays类的static <T> Stream<T> stream(T[] array): 返回一个流
IntStream stream = Arrays.stream(arr);
Employee e1 = new Employee(1001,"Tom");
Employee e2 = new Employee(1002,"Jerry");
Employee[] arr1 = new Employee[]{e1, e2};
Stream<Employee> stream1 = Arrays.stream(arr1);
}
// 创建 Stream方式三:通过Stream的of()
@Test
public void test3() {
Stream<Integer> stream = Stream.of(1, 2, 3, 4, 5, 6);
}
//创建 Stream方式四:创建无限流
@Test
public void test4(){
// 迭代
// public static<T> Stream<T> iterate(final T seed, final UnaryOperator<T> f)
// 遍历前10个偶数
Stream.iterate(0, t -> t + 2).limit(10).forEach(System.out::println);
// 生成
// public static<T> Stream<T> generate(Supplier<T> s)
Stream.generate(Math::random).limit(10).forEach(System.out::println);
}
}
2️⃣步骤二 中间操作
多个中间操作可以连接起来形成一个流水线,除非流水线上触发终止操作,否则中间操作不会执行任何的处理!而在终止操作时一次性全部处理,称为“惰性求值”
筛选与切片
@Test
public void test1() {
List<Employee> list = EmployeeData.getEmployees();
// filter(Predicate p)——接收 Lambda , 从流中排除某些元素。
Stream<Employee> stream = list.stream();
// 练习:查询员工表中薪资大于7000的员工信息
stream.filter(e -> e.getSalary() > 7000).forEach(System.out::println);
System.out.println();
// limit(n)——截断流,使其元素不超过给定数量。
list.stream().limit(3).forEach(System.out::println);
System.out.println();
// skip(n) —— 跳过元素,返回一个扔掉了前 n 个元素的流。
// 若流中元素不足 n 个,则返回一个空流。与 limit(n) 互补
list.stream().skip(3).forEach(System.out::println);
System.out.println();
//distinct()——筛选,通过流所生成元素的 hashCode() 和 equals() 去除重复元素
list.add(new Employee(1010,"刘强东",40,8000));
list.add(new Employee(1010,"刘强东",41,8000));
list.add(new Employee(1010,"刘强东",40,8000));
list.add(new Employee(1010,"刘强东",40,8000));
list.add(new Employee(1010,"刘强东",40,8000));
list.stream().distinct().forEach(System.out::println);
}
映射
@Test
public void test2() {
// map(Function f)——接收一个函数作为参数,将元素转换成其他形式或提取信息,
// 该函数会被应用到每个元素上,并将其映射成一个新的元素。
List<String> list = Arrays.asList("aa", "bb", "cc", "dd");
list.stream().map(str -> str.toUpperCase()).forEach(System.out::println);
// 练习:获取员工姓名长度大于3的员工的姓名。
List<Employee> employees = EmployeeData.getEmployees();
Stream<String> namesStream = employees.stream().map(Employee::getName);
namesStream.filter(name -> name.length() > 3).forEach(System.out::println);
System.out.println();
// flatMap(Function f)
Stream<Stream<Character>> streamStream = list.stream()
.map(StreamAPITest1::fromStringToStream);
streamStream.forEach(s -> {s.forEach(System.out::println);});
System.out.println();
// flatMap(Function f)——接收一个函数作为参数,将流中的每个值都换成另一个流,
// 然后把所有流连接成一个流。
Stream<Character> characterStream = list.stream()
.flatMap(StreamAPITest1::fromStringToStream);
characterStream.forEach(System.out::println);
}
// 将字符串中的多个字符构成的集合转换为对应的Stream的实例
public static Stream<Character> fromStringToStream(String str) { //aa
ArrayList<Character> list = new ArrayList<>();
for (Character c : str.toCharArray()) {
list.add(c);
}
return list.stream();
}
@Test
public void test3() {
ArrayList list1 = new ArrayList();
list1.add(1);
list1.add(2);
list1.add(3);
ArrayList list2 = new ArrayList();
list2.add(4);
list2.add(5);
list2.add(6);
list1.add(list2); // 4个元素,类似map
list1.addAll(list2); // 6个元素,类似flatmap
System.out.println(list1);
}
排序
@Test
public void test4() {
// sorted()——自然排序
List<Integer> list = Arrays.asList(12, 43, 65, 34, 87, 0, -98, 7);
list.stream().sorted().forEach(System.out::println);
// 抛异常,原因:Employee没有实现Comparable接口
// List<Employee> employees = EmployeeData.getEmployees();
// employees.stream().sorted().forEach(System.out::println);
// sorted(Comparator com)——定制排序
List<Employee> employees = EmployeeData.getEmployees();
employees.stream().sorted((e1, e2) -> {
int ageValue = Integer.compare(e1.getAge(), e2.getAge());
if (ageValue != 0) {
return ageValue;
} else {
return -Double.compare(e1.getSalary(), e2.getSalary());
}
}).forEach(System.out::println);
}
3️⃣步骤三 终止操作
终端操作会从流的流水线生成结果。其结果可以是任何不是流的值,例如:List、 Integer,甚至是void
流进行了终止操作后,不能再次使用。
匹配与查找
@Test
public void test1() {
List<Employee> employees = EmployeeData.getEmployees();
// allMatch(Predicate p)——检查是否匹配所有元素。
// 练习:是否所有的员工的年龄都大于18
boolean allMatch = employees.stream().allMatch(e -> e.getAge() > 18);
System.out.println(allMatch);
// anyMatch(Predicate p)——检查是否至少匹配一个元素。
// 练习:是否存在员工的工资大于 10000
boolean anyMatch = employees.stream().anyMatch(e -> e.getSalary() > 10000);
System.out.println(anyMatch);
// noneMatch(Predicate p)——检查是否没有匹配的元素。
// 练习:是否存在员工姓“雷”
boolean noneMatch = employees.stream().noneMatch(e -> e.getName().startsWith("雷"));
System.out.println(noneMatch);
// findFirst——返回第一个元素
Optional<Employee> employee = employees.stream().findFirst();
System.out.println(employee);
// findAny——返回当前流中的任意元素
Optional<Employee> employee1 = employees.parallelStream().findAny();
System.out.println(employee1);
}
@Test
public void test2() {
List<Employee> employees = EmployeeData.getEmployees();
// count——返回流中元素的总个数
long count = employees.stream().filter(e -> e.getSalary() > 5000).count();
System.out.println(count);
// max(Comparator c)——返回流中最大值
// 练习:返回最高的工资:
Stream<Double> salaryStream = employees.stream().map(e -> e.getSalary());
Optional<Double> maxSalary = salaryStream.max(Double::compare);
System.out.println(maxSalary);
// min(Comparator c)——返回流中最小值
// 练习:返回最低工资的员工
Optional<Employee> employee = employees.stream()
.min((e1, e2) -> Double.compare(e1.getSalary(), e2.getSalary()));
System.out.println(employee);
System.out.println();
// forEach(Consumer c)——内部迭代
employees.stream().forEach(System.out::println);
// 使用集合的遍历操作
employees.forEach(System.out::println); // 俩forEach()原理不同
}
归约
备注:map和reduce的连接通常称为 map-reduce模式,因Google用它来进行网络搜索而出名
@Test
public void test3() {
// reduce(T identity, BinaryOperator)——可以将流中元素反复结合起来,得到一个值。返回 T
// 练习1:计算1-10的自然数的和
List<Integer> list = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
Integer sum = list.stream().reduce(0, Integer::sum);
System.out.println(sum);
// reduce(BinaryOperator) ——可以将流中元素反复结合起来,得到一个值。返回 Optional<T>
// 练习2:计算公司所有员工工资的总和
List<Employee> employees = EmployeeData.getEmployees();
Stream<Double> salaryStream = employees.stream().map(Employee::getSalary);
Optional<Double> sumMoney = salaryStream.reduce(Double::sum);
// Optional<Double> sumMoney = salaryStream.reduce((d1, d2) -> d1 + d2);
System.out.println(sumMoney.get());
}
收集
Collector接口中方法的实现决定了如何对流执行收集的操作(如收集到List、Set、Map)
Collectors实用类提供了很多静态方法,可以方便地创建常见收集器实例具体方法与实例如下表
@Test
public void test4(){
// collect(Collector c)——将流转换为其他形式。接收一个 Collector接口的实现,
// 用于给Stream中元素做汇总的方法
// 练习1:查找工资大于6000的员工,结果返回为一个List或Set
List<Employee> employees = EmployeeData.getEmployees();
List<Employee> employeeList = employees.stream().filter(e -> e.getSalary() > 6000)
.collect(Collectors.toList());
employeeList.forEach(System.out::println);
System.out.println("-------------------");
Set<Employee> employeeSet = employees.stream().filter(e -> e.getSalary() > 6000)
.collect(Collectors.toSet());
employeeSet.forEach(System.out::println);
}
Optional类
概述
为了解决java中的空指针问题而生
Optional
Optional类提供的方法
Optional类提供了很多方法,可以不用再现实的进行空值检验
- 创建Optional类对象的方法
**Optional.of(T t)**
创建一个Optional实例,t必须非空;**Optional.empty()**
创建一个空的 Optional 实例**Optional.ofNullable(T t)**
t可以为null
- 判断Optional容器是否包含对象
**boolean isPresent()**
判断是否包含对象**void ifPresent(Consumer<? super T> consumer)**
如果有值,就执行Consumer接口的实现代码,并且该值会作为参数传给它。
- 获取Optional容器的对象
**T get()**
如果调用对象包含值,返回该值,否则抛异常**T orElse(T other)**
如果有值则将其返回,否则返回指定的other对象**T orElseGet(Supplier<? extends t> other)**
如果有值则将其返回,否则返回由Supplier接口实现提供的对象。**T orElseThrow(Supplier<? extends X> exceptionSupplier)**
如果有值则将其返回,否则抛出由Supplier接口实现提供的异常
- 搭配使用
of() 和 get() 方法搭配使用,明确对象非空
ofNullable() 和 orElse() 搭配使用,不确定对象非空
@Test
public void test1() {
Girl girl = new Girl();
// of(T t):保证t是非空的
// girl = null;
Optional<Girl> optionalGirl = Optional.of(girl);
}
@Test
public void test2() {
Girl girl = new Girl();
// ofNullable(T t):t可以为null
girl = null;
Optional<Girl> optionalGirl = Optional.ofNullable(girl);
System.out.println(optionalGirl);
// orElse(T t1):如果当前的Optional内部封装的t是非空的,则返回内部的t.
Girl girl1 = optionalGirl.orElse(new Girl("赵丽颖"));
System.out.println(girl1);
}
public String getGirlName(Boy boy) {
return boy.getGirl().getName();
}
@Test
public void test3() {
Boy boy = new Boy();
// boy = null;
String girlName = getGirlName(boy);
System.out.println(girlName);
}
// 优化以后的getGirlName():
public String getGirlName1(Boy boy) {
if (boy != null) {
Girl girl = boy.getGirl();
if (girl != null) {
return girl.getName();
}
}
return null;
}
@Test
public void test4() {
Boy boy = new Boy();
// boy = null;
String girlName = getGirlName1(boy);
System.out.println(girlName);
}
// 使用Optional类的getGirlName():
public String getGirlName2(Boy boy) {
Optional<Boy> boyOptional = Optional.ofNullable(boy);
// 此时的boy1一定非空
Boy boy1 = boyOptional.orElse(new Boy(new Girl("迪丽热巴")));
Girl girl = boy1.getGirl();
Optional<Girl> girlOptional = Optional.ofNullable(girl);
// girl1一定非空
Girl girl1 = girlOptional.orElse(new Girl("古力娜扎"));
return girl1.getName();
}
@Test
public void test5() {
Boy boy = null;
boy = new Boy();
boy = new Boy(new Girl("苍老师"));
String girlName = getGirlName2(boy);
System.out.println(girlName);
}
public class Boy {
private Girl girl;
public Girl getGirl() {
return girl;
}
public void setGirl(Girl girl) {
this.girl = girl;
}
public Boy() {}
public Boy(Girl girl) {
this.girl = girl;
}
//toString()等略
}