java - peek()和allMatch()如何在Java 8 Stream API中一起工作

标签 java java-8 java-stream

我发现了一个关于peek方法的Java 8 Stream API的测验,如下所示

Arrays.asList("Fred", "Jim", "Sheila")
      .stream()
      .peek(System.out::println)
      .allMatch(s -> s.startsWith("F"));

输出是
Fred
Jim

我对此流的工作方式感到困惑吗?我的预期结果应该是
Fred
Jim
Sheila

peek()方法是一个中间操作,它处理Stream中的每个元素。谁能向我解释一下。

最佳答案

这是流优化,称为短路。本质上,发生的事情是allMatch阻止了对流执行不必要的中间操作,因为在知道最终结果时执行这些操作没有意义。

好像发生了这种情况:

take"Fred"
peek("Fred")
evaluate("Fred".startsWith("F"))
decide whether the result of allMatch() is known for sure: Not yet

take"Jim"
peek("Jim")
evaluate("Jim".startsWith("F"))
decide whether the result of allMatch() is known for sure: Yes

评估"Jim".startsWith("F")时,可以肯定知道allMatch(s -> s.startsWith("F"))的结果。 "Jim"之后的管道中会出现什么值都没有关系,我们知道所有以“F”开头的值都是 false

这不是特定于peek/allMatch组合,存在多个中间和端子短路操作。 java.util.stream package's docs状态:

Further, some operations are deemed short-circuiting operations. An intermediate operation is short-circuiting if, when presented with infinite input, it may produce a finite stream as a result. A terminal operation is short-circuiting if, when presented with infinite input, it may terminate in finite time. Having a short-circuiting operation in the pipeline is a necessary, but not sufficient, condition for the processing of an infinite stream to terminate normally in finite time.



将其扩展到有限的流,并且短路操作避免了不必要的流水线步骤的执行,就像您的示例一样。

关于java - peek()和allMatch()如何在Java 8 Stream API中一起工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60498730/

相关文章:

java - 如何找到并停止所有当前正在运行的线程?

grails - Grails 2.3.x是否支持JDK 8

java - 转换为List后如何迭代方法中的元素?

java - 使用 Java 8 Streams 从不同类型的另一个集合更新集合

java - VSCode getResource 找不到资源

java - 从列表和比较器 java 中删除

java - 确保字符串中包含\n 以特定字符串结尾

java.time 和 JPA

java - Hibernate - Java8 函数对象可以保存在数据库中吗?

java - 使用流生成 short[]