你还在用迭代器处理集合吗?试试Stream,真香!

简介: 首先给大家看一段代码,让大家直观感受下 Java7 和 Java8 遍历处理集合的不同Dish 是一个菜肴对象,calories 属性表示该菜品的卡路里值,name 则是菜品的名称。我们需要过滤出卡路里小于400、然后根据卡路里值升序、接着拿到他们的名称列表并返回

首先给大家看一段代码,让大家直观感受下 Java7 和 Java8 遍历处理集合的不同


Dish 是一个菜肴对象,calories 属性表示该菜品的卡路里值,name 则是菜品的名称。我们需要过滤出卡路里小于400、然后根据卡路里值升序、接着拿到他们的名称列表并返回


Java7

public static List<String> getLowCaloricDishesNamesInJava7(List<Dish> dishes){  
    List<Dish> lowCaloricDishes = new ArrayList<>();  
    for(Dish d: dishes){  
        if(d.getCalories() < 400){  
            lowCaloricDishes.add(d);  
        }  
    }  
    List<String> lowCaloricDishesName = new ArrayList<>();  
    Collections.sort(lowCaloricDishes, new Comparator<Dish>() {  
        public int compare(Dish d1, Dish d2){  
            return Integer.compare(d1.getCalories(), d2.getCalories());  
        }  
    });  
    for(Dish d: lowCaloricDishes){  
        lowCaloricDishesName.add(d.getName());  
    }  
    return lowCaloricDishesName;  
}  

Java8

public static List<String> getLowCaloricDishesNamesInJava8(List<Dish> dishes){  
    return dishes.stream()  
        .filter(d -> d.getCalories() < 400)  
        .sorted(comparing(Dish::getCalories))  
        .map(Dish::getName)  
        .collect(toList());  
}  

如果需要多核并行处理,则只需调用 dishes.parallelStream() 即可


在 Java8 之前,程序员需要通过 2次遍历 + 一次集合排序才能完成的工作,Java8 只需要一个链式调用就可以解决。这就是 Stream 的强大之处


image.png


认识流

流是什么

流是 Java API 的新成员,允许程序员以声明式的方式处理集合数据,并且支持链式调用、支持并行处理。用流处理的集合数据高效且易读。Stream 的 10 种创建方式,这篇推荐你看下。更多Java8新特性教程关注公众号Java技术栈回复Java获取。


流与集合的异同

集合的主要功能是以一定的时间和空间复杂度存储和访问元素,而流主要是用于元素计算


集合中的元素可以随意添加和删除,而流不能添加和删除元素


流的元素是按需计算的,只有当用到时他才会参与计算,而集合中的元素必须提前全都准备好


流只能遍历一次,下面的代码会报错 java.lang.IllegalStateException: stream has already been operated upon or closed 流已经被消费掉

List<String> names = Arrays.asList("Java8", "Lambdas", "In", "Action");  
Stream<String> s = names.stream();  
s.forEach(System.out::println);  
s.forEach(System.out::println);  

集合采用外部迭代,流采用内部迭代。内部迭代意味着 Java 可以替你选择更优的迭代策略和并行处理。而外部迭代如果程序员想着做个更有的迭代/采用并行就相当于“下次一定”😅了

流操作分类

对流的操作可以分为两类,可以继续执行下一个流操作的称为中间操作(方法的返回值是 Stream),关闭流的操作称为终止操作。


中间操作

除非流水线上执行终端操作,否则中间操作不会执行任何处理。流会对中间操作进行合并、短路等优化


终端操作

终端操作会从流的流水线生成结果,返回一个非 stream 的任意类型值


使用流

筛选和切片

筛选

filter(Predicate predicate) 方法可以将流中满足某条件的元素筛选出来。该方法接收一个谓词函数,返回流。比如要选出某个苹果集合中红色的苹果

List<Apple> appleList = new ArrayList<>();  
List<Apple> redAppleList = appleList.stream().filter(a -> "red".equals(a.getColor())).collect(Collectors.toList());  

去重

distinct() 方法会根据元素的 hashCode() 和 equals() 方法对流中元素进行去重操作


截断

limit(n) 方法会返回流的前 n 个元素,对于有序集合List,流会按照添加顺序返回前 n 个元素,而无序集合则不会


跳过

skip(n) 方法会跳过流的前 n 个元素,可以通过 skip(m).limit(n) 返回列表中第 m - (m+n) 区间的元素,类似与 mysql 中的 limit m,n


映射

对流中的每个元素应用函数

map(Function mapper) 方法。该方法接收一个 Function 函数,对流中的每一个元素使用。然后可以返回任意类型的对象。有了该方法,就可以结合 Lambda 表达式对集合中的元素使用函数进行各种转换


流的扁平化

flatMap() 可以将流操作中多个流合并成一个流的多个元素。


举个例子:集合 words 有两个单词,现在想获得[H, e, l, o, W, r, d] 在 split 方法执行完毕后,返回的是 Stream(String[]) 对象,而此时如果执行 map 方法,返回的就是多个流的集合(这个例子中就是两个 Stream(String)),这时是无法继续接下来的 distinct 操作的,因此需要 flatMap 将两个 Stream扁平化成一个 Stream,然后进行操作

List<String> words = Arrays.asList("Hello", "World");  
List<String> charList = words.stream().map(word -> word.split("")).flatMap(Arrays::stream).distinct().collect(Collectors.toList());  

该方法的方法声明 flatMap(Function> mapper) 中可以看出,他所使用的函数式接口 Function 第二个泛型 R 必须是 Stream流。即函数式接口的抽象方法返回值必须是 Stream流及其子类对象。


查找和匹配

检查谓词是否至少匹配一个元素

anyMatch 方法可以回答“流中是否存在至少一个复合谓词条件的元素”返回 boolean 类型的值,因此是一个终端操作,例如

List<Integer> num = Arrays.asList(1, 2, 3, 4, 5, 6);  
if (num.stream().anyMatch(n -> n % 3 == 0)) {  
    System.out.println("集合中有元素是3的整数倍");  
}  

控制台会输出'集合中有元素是3的整数倍',因为集合中 3、6都是3的整数倍,符合谓词的条件。


检查谓词是否匹配所有元素

allMatch 方法和 anyMatch 方法原理类似,但是它仅当所有元素满足谓词条件时,返回 true。


noneMatch 与 allMatch 正好相反,仅当所有元素不满足谓词条件时,返回 true


ps:和 && || 运算符类似,以上三个操作都用到了短路的思想来提高效率。


查找元素

findAny() 该方法返回当前流中的任意元素,可以和其他流操作结合使用,这里需要注意 findAny() 返回的结果被 Optional 所包裹,Optional是 Java8 为优雅的避免 NPE 所采用的新 API,这里需要说明的就是 Optional.ifPresent(Consumer consumer) 表示当 Optional 包裹的元素不为空时,执行 consumer

num.stream().filter(n -> n > 2).findAny().ifPresent(System.out::println);  

findFirst() 该方法返回当前流中的第一个元素,一般也和其他流操作(例如 filter() 过滤)结合使用。与 findAny() 不同的是,他一定返回有序集合的第一个满足条件的元素。当然有得必有失,作为代价,findFirst() 在并行处理时限制更多一些。


归约

元素求和

reduce(T identity, BinaryOperator accumulator); 方法接收两个参数:identity 初始值,accumulator 对两个数的操作。例如求集合中数字的和:

num.stream().reduce(0, (a, b) -> a + b) // 计算完成,返回 21  

ps:Lambda 表达式 (a, b) -> a + b) 中 a 是上一轮执行完后的累计值,b 是本次循环流中的元素。通过累加就可以计算出数字的和。更多Java8新特性教程关注公众号Java技术栈回复Java获取。


最大值和最小值

reduce 方法不仅可以求和、求积。甚至可以计算最大值、最小值。

num.stream().reduce(Integer::max);  
num.stream().reduce(Integer::min);  

总结

流是 Java API 的新成员,允许程序员以声明式的方式处理集合数据,并且支持链式调用、支持并行处理。用流处理的集合数据高效且易读。


流的API中可以分为两大类,中间操作和终端操作,中间操作返回流对象,可以链式调用,终端操作则返回非流对象。


流提供了很多方便的API,如筛选 filter、去重 distinct、截断 limit、跳过 skip、函数转换 map、扁平化 flatMap、判断流中是否有任意元素符合要求 anyMatch、是否所有元素都符合要求 allMatch、是否所有元素都不符合要求 noneMatch、查找元素 findAny findFirst、累计式的计算元素 reduce


相关文章
|
NoSQL MongoDB 数据库
MongoDB 解析:灵活文档数据库与 Docker Compose 部署
`MongoDB` 是一款开源、高性能的 `NoSQL` 数据库,以其无模式的文档存储格式(BSON)而著称,广泛应用于众多开源项目,包括但不限于 Yapi 等。它在大规模数据存储和实时数据处理方面表现出色,因此备受青睐。在本文中,我们将深入探讨 `MongoDB` 的特性,并详细阐述如何使用 Docker Compose 轻松部署 `MongoDB` 数据库,为你提供全方位的指导。
688 1
MongoDB 解析:灵活文档数据库与 Docker Compose 部署
|
安全 网络安全 定位技术
华为基础数通知识
了解基本的数通知识,成为更好的自己
|
存储 弹性计算 移动开发
玩机教程:阿里云电脑无影怎么样?怎么使用?
玩机教程:阿里云电脑无影怎么样?怎么使用?
1198 0
|
人工智能 分布式计算 大数据
开源大数据平台 3.0 技术解读
阿里云研究员,阿里云计算平台事业部开源大数据平台负责人王峰围绕新一代的流式湖仓、全面 Serverless 化、更智能的开源大数据等多维度解读开源大数据平台 3.0~
1781 1
开源大数据平台 3.0 技术解读
|
消息中间件 关系型数据库 MySQL
Flink最后一站___Flink数据写入Kafka+从Kafka存入Mysql
Flink最后一站___Flink数据写入Kafka+从Kafka存入Mysql
319 0
|
10月前
|
应用服务中间件 Linux nginx
部署使用 CHAT-NEXT-WEB 基于 Deepseek
本文介绍如何在阿里云轻量服务器上部署基于 `Deepseek` 的 `CHAT-NEXT-WEB` 项目。首先,准备一台 Linux 服务器并安装 Docker,确保防火墙允许特定端口访问。接着,通过阿里云容器镜像服务解决国内网络限制问题,将镜像推送到私有仓库并拉取到本地。配置并启动 `chat-next` 项目,使用 Deepseek API 进行优化。最后,安装 Nginx 和 Certbot 配置 HTTPS 访问,确保安全性和自动续签。整个过程需严格遵循官方文档,以避免因网络问题导致的安装失败。
|
编解码 开发工具 Android开发
Android平台GB28181设备接入侧如何实现GB28181-2022实时快照
GB/T28181-2022标准中明确了快照的具体要求,包括图像抓拍配置命令的发送与接收流程。源设备需向目标设备发送包含传输路径和会话ID等信息的命令,目标设备完成图像传输后,通过IETF RFC 3428中的MESSAGE方法发送图像抓拍传输完成的通知。图像格式推荐使用JPEG,且分辨率应与主码流相同。技术实现上,如使用大牛直播SDK在Android平台上,可通过创建`SnapShotImpl`类并调用`capture()`方法实现快照功能,之后将JPEG格式的快照文件上传至国标平台。
333 2
|
机器学习/深度学习 计算机视觉
TPAMI 2024:计算机视觉中基于图神经网络和图Transformers的方法和最新进展
【10月更文挑战第3天】近年来,图神经网络(GNNs)和图Transformers在计算机视觉领域取得显著进展,广泛应用于图像识别、目标检测和场景理解等任务。TPAMI 2024上的一篇综述文章全面回顾了它们在2D自然图像、视频、3D数据、视觉与语言结合及医学图像中的应用,并深入分析了其基本原理、优势与挑战。GNNs通过消息传递捕捉非欧式结构,图Transformers则结合Transformer模型提升表达能力。尽管存在图结构构建复杂和计算成本高等挑战,但这些技术仍展现出巨大潜力。论文详细内容见:https://arxiv.org/abs/2209.13232。
727 3
|
SQL Oracle Java
SQL中使用NEXTVAL获取序列值
SQL中使用NEXTVAL获取序列值
|
缓存 前端开发 JavaScript
彻底学会react hooks API以及应用场景
【4月更文挑战第6天】 React Hooks是16.8版引入的新特性,允许在函数组件中使用state和其它React功能,避免类组件。主要包括useState(添加state)、useEffect(处理副作用)、useContext(访问上下文)、useReducer(使用reducer)、useCallback(缓存函数)、useMemo(缓存计算结果)和useRef(访问DOM元素)。Hooks简化了代码组织和重用,提高了代码效率和可维护性。
340 0