学习札记-Java8系列-9-再次理解Stream
Stream的定义
上一篇我们已经初步体验了Stream的使用,那么,Stream到底是什么呢?简短的定义就是“从支持数据处理操作的源生成的元素序列”。下面来对这句定义中的每一个关键词进行解释
●元素序列——就像集合一样,流也提供了一个接口,可以访问特定元素类型的一组有序值。因为集合是数据结构,所以它的主要目的是以特定的时间/空间复杂度存储和访问元素(如ArrayList 与 LinkedList)。但流的目的在于表达计算,比如filter、sorted和map。集合讲的是数据,流讲的是计算。
●源——流会使用一个提供数据的源,如集合、数组或输入/输出资源。 请注意,从有序集合生成流时会保留原有的顺序。由列表生成的流,其元素顺序与列表一致。
●数据处理操作——流的数据处理功能支持类似于数据库的操作,以及函数式编程语言中的常用操作,如filter、map、reduce、find、match、sort等。流操作可以顺序执行,也可并行执行。
此外,流操作有两个重要的特点。
●流水线——很多流操作本身会返回一个流,这样多个操作就可以链接起来,形成一个大的流水线。流水线的操作可以看作对数据源进行数据库式查询。
●内部迭代——与使用迭代器外部迭代的集合不同,流的迭代操作是在背后进行的。
--摘自《Java 8 In Action》
■总结:整体来看,流式思想类似于车间的“生产流水线”。
图中展示了过滤、映射、跳过、计数等多步操作,每一个竖着的方框都是一个“流”,调用指定的方法,可以从一个“流”转换为另一个“流”。
内部迭代和外部迭代
使用iterator迭代器迭代集合的方式,称之为外部迭代,说的通俗一点,就是需要我们程序猿手动的对这个集合进行种种操作才能得到想要结果的迭代方式,叫做外部迭代。
与外部迭代所对应的,则是内部迭代,内部迭代与之相反,是集合本身内部通过流进行了处理之后,程序猿们只需要直接取结果就行了,这种迭代称为内部迭代。
图解:
代码演示:
@Test
public void test2() throws Exception {
System.out.println("=======================外部迭代=========================");
List<Integer> list = Arrays.asList(1, 2, 1, 3, 3, 2, 4);
for (Integer integer : list) {
System.out.println(integer);
}
Iterator<Integer> iterator = list.iterator();
while (iterator.hasNext()) {
System.out.println(iterator.next());
}
System.out.println("=======================内部迭代=========================");
Stream<Integer> stream = list.stream().filter(i -> {
System.out.println("内部迭代惰性求值");
return i % 2 == 0;
});
stream.forEach(System.out::println);//只有执行终止操作的时候才会真正执行中间操作(惰性求值)
}