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方式一:通过集合@Testpublic 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方式二:通过数组@Testpublic 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()@Testpublic void test3() {Stream<Integer> stream = Stream.of(1, 2, 3, 4, 5, 6);}//创建 Stream方式四:创建无限流@Testpublic 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️⃣步骤二 中间操作
多个中间操作可以连接起来形成一个流水线,除非流水线上触发终止操作,否则中间操作不会执行任何的处理!而在终止操作时一次性全部处理,称为“惰性求值”
筛选与切片

@Testpublic 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);}
映射

@Testpublic 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) { //aaArrayList<Character> list = new ArrayList<>();for (Character c : str.toCharArray()) {list.add(c);}return list.stream();}@Testpublic 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个元素,类似maplist1.addAll(list2); // 6个元素,类似flatmapSystem.out.println(list1);}
排序

@Testpublic 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
流进行了终止操作后,不能再次使用。
匹配与查找


@Testpublic void test1() {List<Employee> employees = EmployeeData.getEmployees();// allMatch(Predicate p)——检查是否匹配所有元素。// 练习:是否所有的员工的年龄都大于18boolean allMatch = employees.stream().allMatch(e -> e.getAge() > 18);System.out.println(allMatch);// anyMatch(Predicate p)——检查是否至少匹配一个元素。// 练习:是否存在员工的工资大于 10000boolean 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);}@Testpublic 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用它来进行网络搜索而出名
@Testpublic 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实用类提供了很多静态方法,可以方便地创建常见收集器实例具体方法与实例如下表

@Testpublic void test4(){// collect(Collector c)——将流转换为其他形式。接收一个 Collector接口的实现,// 用于给Stream中元素做汇总的方法// 练习1:查找工资大于6000的员工,结果返回为一个List或SetList<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() 搭配使用,不确定对象非空
@Testpublic void test1() {Girl girl = new Girl();// of(T t):保证t是非空的// girl = null;Optional<Girl> optionalGirl = Optional.of(girl);}@Testpublic void test2() {Girl girl = new Girl();// ofNullable(T t):t可以为nullgirl = 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();}@Testpublic 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;}@Testpublic 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();}@Testpublic 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()等略}
