当前位置: 首页 > 知识库问答 >
问题:

Stream.foreach是否尊重顺序流的相遇顺序?

萧鹏云
2023-03-14

stream.foreach的Javadoc表示(强调是我的):

此操作的行为显式不确定。对于并行流管道,此操作不能保证尊重流的相遇顺序,因为这样做会牺牲并行性的好处。对于任何给定的元素,操作可以在库选择的任何时间和线程中执行。如果操作访问共享状态,则它负责提供所需的同步。

同样的文本也出现在Java9早期访问Javadoc中。

for (Foo foo : collection)
    foo.bar();
collection.stream().forEach((foo) -> {
    foo.bar();
});

如果forEach不保留遭遇顺序,则会引入bug。在报告针对NetBeans的bug之前,我想知道库实际上保证了什么,并由源代码进行了备份。

我在寻找一个来自权威来源的答案。这可能是库实现中的明确注释、Java开发邮件列表上的讨论(Google没有为我找到任何东西,但可能我不知道这些神奇的词),或者是库设计者的声明(我知道其中两个设计者,Brian Goetz和Stuart Marks在Stack Overflow上是活跃的)。(请不要回答“只使用forEachOrdered”--我已经使用了,但我想知道不使用的代码是否错误。)

共有1个答案

卓瀚
2023-03-14

规范的存在是为了描述调用者可以依赖的最小保证,而不是描述实现的功能。这一差距是至关重要的,因为它允许实现灵活性的发展。(规范是声明性的;实现是必要的。)规格过高和规格不足一样糟糕。

当一个规范说“不保存属性X”时,这并不意味着属性X可能永远不会被观察到;这意味着实现没有义务保存它。你声称的遭遇顺序永远不会被保留的暗示只是一个错误的结论。(hashset并不保证迭代它的元素会保留它们插入的顺序,但这并不意味着这不会意外发生--您只是不能指望它。)

类似地,您的含义“这表明forEach旨在为顺序流保留顺序”,因为您在某些情况下看到了这样做的实现,这同样是不正确的。

 类似资料:
  • 问题内容: Javadoc 表示(重点是我): 该操作的行为明确地是不确定的。 对于并行流管道,此操作不能保证遵守流的遇到顺序 ,因为这样做会牺牲并行性的好处。对于任何给定的元素,可以在库选择的任何时间和线程中执行操作。如果操作访问共享状态,则它负责提供所需的同步。 Java 9 Early Access Javadoc中提供了相同的文本。 第一句话(“明确地不确定”)表明(但未明确说明)此方法未

  • 的JavaDocs说,如果流有遇到顺序,那么第一个元素将始终返回,但如果流没有遇到顺序,则可以返回任何元素。 我试图演示这如何在没有遭遇顺序的流上工作,但是除了实际的第一个元素之外,我不能让它返回任何东西。 我尝试将元素添加到< code >集合中,该集合没有已定义的相遇顺序: 每次我跑步时,我都会得到作为第一个字符串。然后,我尝试在将列表添加到之前在上进行 ,但这并没有改变任何内容。 我还是每次

  • 是否有任何保证在顺序和有序流上的操作是按遇到顺序处理的? 我是说,如果我有这样的代码: 是否可以保证它将按照生成范围的遇到顺序执行myFunction()调用? 我找到了Stream类的JavaDocs草案,它明确地说明了以下内容: 对于顺序流管道,如果管道源具有已定义的遇到顺序,则所有操作都按照管道源的遇到顺序执行。 但是它没有提到顺序流,这个例子是针对并行流的(我的理解是,顺序流和并行流都是正

  • 问题内容: 我认为使用某种顺序才有意义。我想做的是在视图中包括该子句,以便该视图上的所有s都可以忽略它。但是,我担心该订单不一定会延续到,因为它没有指定订单。 是否存在一种情况,即视图指定的顺序不会反映在该视图上的select结果中(该视图中的order by子句除外)? 问题答案: 您不能指望没有显式子句的任何查询中的行顺序。如果查询有序视图,但没有包括子句,则如果它们的顺序正确,请感到惊喜,并

  • 我有一个记录课程: 我创建了一个包含很多记录的大列表。只有第二个和第五个值,即i/10000和i,稍后分别由getter使用。 请注意,前10000条记录的类别2为0,接下来的10000条记录的类别1等,而值1按顺序为0-114999。 我创建了一个既并行又排序的流。 我有一个ForkJoinPool,它维护8个线程,这是我电脑上的内核数。 我使用这里描述的技巧将流处理任务提交给我自己的,而不是常