## Stream 流式操作 ### 概述 ​ java 8 是一个非常成功的版本,这个版本新增的Stream,配合同版本出现的Lambda ,给我们操作集合(Collection)提供了极大的便利。Stream流是JDK8新增的成员,允许以声明性方式处理数据集合,可以把Stream流看作是遍历数据集合的一个高级迭代器。Stream 是 Java8 中处理集合的关键抽象概念,它可以指定你希望对集合进行的操作,可以执行非常复杂的查找/筛选/过滤、排序、聚合和映射数据等操作。使用Stream API 对集合数据进行操作,就类似于使用 SQL 执行的数据库查询。也可以使用 Stream API 来并行执行操作。简而言之,Stream API 提供了一种高效且易于使用的处理数据的方式。 ​ 可以把Stream当作是一个工厂,我们要操作的集合数据作为原材料,原材料通过我们调用Stream的方法加工成所需的成品供我们使用。 ### Stream类型 ​ **jdk8提供了BaseStream流的基本接口,以及四种实现了BaseStream的Stream接口** | 接口 | 类型 | 概述 | | ------------------------------------------------------------ | ---- | -------------------------------------------------------- | | interface BaseStream> | | Stream流顶级接口所有的Stream都是基于BaseStream进行实现的 | | interface Stream<T> extends BaseStream | 通用 | 支持顺序和并行操作的任意类型的元素流 | | interface DoubleStream extends BaseStream | 数值 | 支持顺序和并行聚合操作的原始 double 值的元素流 | | interface IntStream extends BaseStream | 数值 | 支持顺序和并行聚合操作的原始 int 值的元素流 | | interface LongStream extends BaseStream | 数值 | 支持顺序和并行聚合操作的原始 long 值的元素流 | ​ **BaseStream 抽象方法介绍** | 方法 | 类型 | 概述 | | -------------------------------- | -------- | ------------------------------------------------------------ | | Iterator iterator() | 终端操作 | 返回此流的元素的迭代器 | | Spliterator spliterator() | 终端操作 | 返回此流的元素的拆分器 | | boolean isParallel() | | 检测stream是否为并行stream(执行过终端操作方法的流在执行此方法会出现不可预知的问题) | | S sequential() | 中间操作 | 返回一个顺序流 | | S parallel() | 中间操作 | 返回一个并行流 | | S unordered() | 中间操作 | 返回一个无序流 | | S onClose(Runnable closeHandler) | 中间操作 | 返回具有附加关闭处理程序的等效流 | | void close() | | 关闭此流,从而调用此流管道的所有关闭处理程序 | > 数值类型的Stream流额外提供了sum、average、boxed等方法,sum方法用于求和、average用于求平均值,boxed方法会将数值流转换为通用流; > > **下面我会以通用Stream流来做讲解** ### Stream的创建方式 **下面我举例常用Stream流创建的方式,其中加粗的方法是用到最多的几种.** | 创建方法 | 方法类型 | 顺序流/并行流 | | ----------------------------- | -------- | ------------- | | **Stream.of** | 静态方法 | 顺序流 | | Stream.generate | 静态方法 | 顺序流 | | Stream.builder | 静态方法 | 顺序流 | | Stream.iterate | 静态方法 | 顺序流 | | **Arrays.stream** | 静态方法 | 顺序流 | | **Collection.stream** | 成员方法 | 顺序流 | | **Collection.parallelStream** | 成员方法 | 并行流 | > 并行流:内部以多线程并行执行的方式对流进行操作,因为是多线程并行操作所以无法保证集合数据的顺序,除了无法保证顺序以外还会有线程不安全的问题。 **案例演示** ```java @Test public void test1(){ //通过Stream.of静态方法创建顺序流 Stream stream1 = Stream.of(1, 2, 3); System.out.print("stream1:"); stream1.forEach(System.out::print); //通过Stream.generate静态方法创建顺序流 Stream stream2 = Stream.generate((Math::random)).limit(2); System.out.print("\nstream2:"); stream2.forEach(System.out::print); //通过Stream.builder静态方法创建顺序流 Stream stream3 = Stream.builder() .add(4) .add(5) .add(6).build(); System.out.print("\nstream3:"); stream3.forEach(System.out::print); //通过Stream.iterate静态方法创建顺序流 Stream stream4 = Stream.iterate(0, (x) -> x + 10).limit(5); System.out.print("\nstream4:"); stream4.forEach(System.out::print); //通过Arrays.stream静态方法创建顺序流 Stream stream5 = Arrays.stream(new Integer[]{7, 8, 9}); System.out.print("\nstream5:"); stream5.forEach(System.out::print); Collection collection = new ArrayList<>(); collection.add("大山"); collection.add("大海"); collection.add("大地"); //Collection.stream方法创建顺序流 Stream stream6 = collection.stream(); System.out.print("\nstream6:"); stream6.forEach(System.out::print); //通过Collection.parallelStream创建并行流 Stream stream7 = collection.parallelStream(); System.out.print("\nstream7:"); stream7.forEach(System.out::print); } ``` **输出内容** ``` stream1:123 stream2:0.63071892380716040.5348826766523751 stream3:456 stream4:010203040 stream5:789 stream6:大山大海大地 stream7:大海大地大山 ``` ### Stream方法类型以及方法概述 ### Stream方法案例演示 ### Stream并行流 ### Collectors