zoukankan      html  css  js  c++  java
  • Java8中findAny和findFirst的区别

    试验了一下java stream中的findAny和findFirst,发现都返回了列表中的第一个元素。那么,这两种方法的区别是什么呢?

    查看了一下Java API document:

    findFirst:

    findFirst

    Optional<T> findFirst()

    Returns an Optional describing the first element of this stream, or an empty Optional if the stream is empty. If the stream has no encounter order, then any element may be returned.

    This is a short-circuiting terminal operation.

    Returns:

    an Optional describing the first element of this stream, or an empty Optional if the stream is empty

    Throws:

    NullPointerException - if the element selected is null

    顾名思义,即返回列表中的第一个元素。

    这里的short-circuiting是指:有时候需要在遍历中途停止操作,比如查找第一个满足条件的元素或者limit操作。在Stream中short-circuiting操作有:anyMatch、allMatch、noneMatch、findFirst、findAny、limit,这些操作在Sink中都有一个变量来判断是否短路,比如limit用的是m,match用的是stop,find用的是hasValue。

    这里的terminal operation是指:一个终结操作,比如foreach,IntStream.sum

    那么,findAny是什么呢?

    findAny

    Optional<T> findAny()

    Returns an Optional describing some element of the stream, or an empty Optional if the stream is empty.

    This is a short-circuiting terminal operation.

    The behavior of this operation is explicitly nondeterministic(不确定的); it is free to select any element in the stream. This is to allow for maximal performance in parallel operations; the cost is that multiple invocations(调用) on the same source may not return the same result. (If a stable result is desired, use findFirst() instead.)

    Returns:

    an Optional describing some element of this stream, or an empty Optional if the stream is empty

    Throws:

    NullPointerException - if the element selected is null

    See Also:

    findFirst()

    可以看到findAny()操作,返回的元素是不确定的,对于同一个列表多次调用findAny()有可能会返回不同的值。使用findAny()是为了更高效的性能。如果是数据较少,串行地情况下,一般会返回第一个结果,如果是并行的情况,那就不能确保是第一个。比如下面的例子会随机地返回OptionalInt[50]。

    System.out.println(IntStream.range(0, 100).parallel().findAny());

    让我们来举另外一个例子:

    1.  
      List<String> lst1 = Arrays.asList("Jhonny", "David", "Jack", "Duke", "Jill","Dany","Julia","Jenish","Divya");
    2.  
      List<String> lst2 = Arrays.asList("Jhonny", "David", "Jack", "Duke", "Jill","Dany","Julia","Jenish","Divya");
    3.  
       
    4.  
      Optional<String> findFirst = lst1.parallelStream().filter(s -> s.startsWith("D")).findFirst();
    5.  
      Optional<String> fidnAny = lst2.parallelStream().filter(s -> s.startsWith("J")).findAny();
    6.  
       
    7.  
      System.out.println(findFirst.get()); //总是打印出David
    8.  
      System.out.println(fidnAny.get()); //会随机地打印出Jack/Jill/Julia
  • 相关阅读:
    编写测试类实现并发访问固定URL(亲测能用!!!)
    java项目添加log4j打印日志+转换系统时间
    springboot项目没错,但就是报红叉
    我想查看数据库名,输入命令:select name from v$database;为什么会说表和视图不存在
    DRUID连接池的实用 配置详解+使用方法+监控方式(太强大了!!!)
    Druid连接池 属性说明
    springBoot2.2.0+mybatis-xml文件方式+Oracle11g+jsp页面,实现简单的CRUD
    s5-12 RIP
    s5-12 RIP
    s5-13 RIP 为什么会 衰败
  • 原文地址:https://www.cnblogs.com/ylsx/p/14103694.html
Copyright © 2011-2022 走看看