【问题标题】:Java 8 : Stream.filter not running as expectedJava 8:Stream.filter 未按预期运行
【发布时间】:2021-04-05 17:18:25
【问题描述】:

给定:

    List<String> str = Arrays.asList ("my", "pen", "is", "your", "pen");
    Predicate<String> test = s -> {
        int i = 0;
        boolean result = s.contains ("pen");
        System.out.print((i++) + ":");
        return result;
    };

这打印:0:0:

    str.stream().filter(test).findFirst();

这会打印 0:0:0:0:0:

    str.stream().filter(test).collect(Collectors.toList());

让我困惑的是,无论是findFirst还是collect,无论终端操作是否短路,他们两者都应该遍历列表中的每个项目,对吧?

那么为什么在第一个示例 findFirst 中,“0:”被打印两次,而不仅仅是一次,而不是5 次? p>

【问题讨论】:

  • 当 API 读取“find first”时,为什么会期望它进一步迭代?
  • 您的计数打印使用的是后增量,并且始终为零。

标签: java java-8 java-stream short-circuiting


【解决方案1】:

让我困惑的是,无论是findFirst还是collect,无论终端操作是否短路,它们都应该遍历列表中的每一项,对吧?

能够在不检查整个流的情况下返回正是导致操作(如findFirst)短路的原因。见Streams javadoc

Streams 文档实际上并没有承诺如何或是否评估您的谓词。它只是保证它将返回与谓词匹配的流的第一个元素。

【讨论】:

    【解决方案2】:

    第一个Stream的思想是迭代找到第一次出现的“pen”,所以Predicate迭代数组直到找到第一个匹配过滤条件的元素。因此,您只会看到两个“0:”。 Official documentation

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2015-10-13
      • 2014-11-12
      • 1970-01-01
      • 2020-06-28
      • 2012-02-18
      • 2018-01-18
      • 2012-06-14
      • 2019-03-03
      相关资源
      最近更新 更多