【学习笔记】Java函数式编程03 Stream流-终结操作
书接上回
3.3.3 终结操作
3.3.3.1 forEach
对集合的每一个元素进行处理
接触很多了不赘述
3.3.3.2 count
用来获取当前流中的元素的个数
比如,打印出所有作家的作品的总数
System.out.println(authors.stream().flatMap(author -> author.getBooks().stream()).count());
3.3.3.3 max和min
可以用来流中的最值。
查看源码可以发现这个方法是需要传入参数的
Optional<T> max(Comparator<? super T> comparator);
Optional<T> min(Comparator<? super T> comparator);
Comparator是一个比较器,但是这里的比较器是什么作用呢?其实stream流的球最值和前面的sorted的实现逻辑类似,都是要求先比较排序后再求最值
另外,这个方法的返回值是Optional类型,这个类型后续会再细讲。
所以max和min不用死记,学会sorted就可以满足需要了
3.3.3.4 ⭐️collect
将流当中的元素转换为一个集合(List/Set/Map)。
阅读源码可以发现collect的两个参数都非常复杂,在实际使用中会使用工具类的方法来快速实现

小练习快速上手
1、获取一个所有作者名字的集合List
authors.stream().map(Author::getName).collect(Collectors.toList())
2、获取一个所有书名的集合SET
authors.stream().flatMap(author -> author.getBooks().stream()).map(Book::getName).collect(Collectors.toSet());
顺带一提,由于set的去重作用,所有其实前面提到的“去重“的需求,也可以尝试用collect来实现
3、获取一个map集合,Key为作者名,Value为作品集List
沿用前面的思路,利用Collectors.toMap方法实现,但是转map是需要指定key和v的,研究toMap方法就可以发现:这里确实要求传入两个Function函数式接口,如下

尝试先写一个匿名内部类的原生写法,然后进行简化:
// 获取一个map集合,Key为作者名,Value为作品集List
List<Author> authors = Author.getAuthor(5);
Map<String, List<Book>> map = authors.stream().collect(Collectors.toMap(new Function<Author, String>() {@Overridepublic String apply(Author author) {return author.getName();}}, new Function<Author, List<Book>>() {@Overridepublic List<Book> apply(Author author) {return author.getBooks();}}));System.out.println(map);// 第一次简化
authors.stream().collect(Collectors.toMap(author -> author.getName(), author -> author.getBooks()));// IDEA提供的进一步简化
authors.stream().collect(Collectors.toMap(Author::getName, Author::getBooks));
不难发现,toMap方法就是分别指定key和value就可以实现转map的要求了
4、获取一个map集合,Key为作者名,Value为作者
注意:需求不同,这次是将val作为作者本身的对象,这个其实也是比较常见的需求
Map<String, List<Author>> collect = author.stream().collect(Collectors.groupingBy(new Function<Author, String>() {@Overridepublic String apply(Author author) {return author.getName();}}));// 简化后
author.stream().collect(Collectors.groupingBy(Author::getName));
介绍完简单的终结操作后,接下来还有一些比较复杂的操作。比如需求场景
- 判断是否有25岁以上的作家?
- 判断是不是所有作家都刚满18岁?
3.3.4 终结操作之查找与匹配
3.3.4.1 anyMatch
有任一元素符合条件,则返回true
3.3.4.2 allMatch
所有符合条件,则返回true
3.3.4.3 nonMatch
所有的元素都不符合条件
(这个不需要硬记,和allMatch互为补集)
3.3.4.4 findAny

这个操作用于获取流中的任意一个元素,注意,这个操作无法保证获取的是第一个元素
3.3.4.5 findFirst
这个操作用于获取流中的第一个元素。
- 这个终结操作一般需要结合
limit和sorted使用
补充说明
这两个方法返回的对象是JDK8新增的Optional对象,用于避免空指针等异常的,后续会详细介绍。
- 要获取里面的对象使用get()方法即可。
- ifPresent()方法可以在对象存在时,执行方法内的函数
3.3.5 ⭐️终结操作之归并操作-reduce
reduce单词字面意思有减少的意思,可以引申为缩小、裁剪的意思
3.3.5.1 reduce概念
对流中的数据,按照指定的计算方式,计算出一个结果。(缩紧操作)
reduce的作用,将stream中的元素“组合”起来,最终传出组合的结果,起到一个紧缩、简化的作用。
两种实现方式
- 传入一个初始值,按照给定的计算方式以此与流中的每个元素进行“计算”,每次计算得到的结果都可以和后面的元素再进行计算,并最终给出结果。
- 没有初始值,而是将第一个元素给定为初始值,然后以此和流内的其他元素进行“计算”并给出结果。
两种方式对应reduce的两种重写方式
T reduce(T identity, BinaryOperator<T> accumulator);Optional<T> reduce(BinaryOperator<T> accumulator);
3.3.5.2 reduce有初始值的重写
T reduce(T identity, BinaryOperator<T> accumulator);
查看源码注释可以发现,双参数的实现逻辑如下:
T result = identity;
for (T element : this stream) result = accumulator.apply(result, element) return result;
做两个求最值的快速练习——求出所有作家的最大年纪
List<Author> authors = Author.getAuthor();
// 先打印出每个作家的年纪
authors.stream().map(Author::getAge).sorted().forEach(System.out::println);
System.out.println("-----------------");
// 求最大值 匿名内部类写法
Integer max = authors.stream().map(Author::getAge).reduce(Integer.MIN_VALUE, new BinaryOperator<Integer>() {@Overridepublic Integer apply(Integer element, Integer next) {return element < next ? next : element;}});
System.out.println(max);
// 求最大值 lambda简化
Integer max1 = authors.stream().map(Author::getAge).reduce(Integer.MIN_VALUE, (element, next) -> element < next ? next : element);
System.out.println(max1);

顺带一提前面学到的max()和min()底层就是使用reduce实现的
3.3.5.3 reduce无初始值的调用
Optional<T> reduce(BinaryOperator<T> accumulator);
实现低层逻辑是这样的:
boolean foundAny = false;
T result = null;
for (T element : this stream) { if (!foundAny) { foundAny = true; result = element; } else result = accumulator.apply(result, element);
}
return foundAny ? Optional.of(result) : Optional.empty();
逻辑就是:
- 第一个元素给定为初始值,然后以此和流内的其他元素进行“计算”并给出结果。
- 由于如果流为空返回的对象可能为空,所有这里使用了Optional进行包装
做个练习,一样是求最大年纪
// 匿名内部类原生写法
Optional<Integer> optional = authors.stream().map(Author::getAge).reduce(new BinaryOperator<Integer>() {@Overridepublic Integer apply(Integer element, Integer next) {return element < next ? next : element;}});
// lambda简化
Optional<Integer> optional1 = authors.stream().map(Author::getAge).reduce((element, next) -> element < next ? next : element);
optional.ifPresent(System.out::println);
optional1.ifPresent(System.out::println);
3.4 Stream流的注意事项
- 惰性求值
- 一次性流
- 不会影响原数据
3.4.1 惰性求值
没有任何终结操作,前面的中间操作都不会得到执行和保留。
- 实际开发过程中,由于没有终结操作的stream写法并不会编译报错
- 所以在写代码的时候一定要养成添加终结操作的习惯)
3.4.2 流是一次性的
在进行终结操作后,流会失效(报废。
举个例子:

- 在实际开发过程中,使用stream流应该在调用stream()方法后就使用链式编程直到终结操作
- 如果需要再次操作,就重新调用并生成新的流即可
3.4.3 不会影响原数据
特指是正常操作,而且这也是选择stream流所期望的。
举个例子,在map操作中将年龄+10,其实集合中元素的年龄是不会变化的。

- 在实际开发中,如果是需要对集合的元素进行操作时,则不建议使用stream流
- 使用了stream流也应该尽量避免对集合中的元素进行操作(增删改)
相关文章:
【学习笔记】Java函数式编程03 Stream流-终结操作
书接上回 3.3.3 终结操作 3.3.3.1 forEach 对集合的每一个元素进行处理 接触很多了不赘述 3.3.3.2 count 用来获取当前流中的元素的个数 比如,打印出所有作家的作品的总数 System.out.println(authors.stream().flatMap(author -> author.getBooks().stre…...
2024 Android保活总结
本文介绍的方案都是无需用户主动开启权限的。如果需要用户主动开启权限或者加白名单之类的话保活的意义就不大了,毕竟用户不大可能主动原因让app一直在后台运行 常规的方案 OnePixelActivity 1,监听SCREEN_OFF启动一个像素的Activity,灭屏…...
迅为RK3568开发板Ubuntu上使用串口调试
我们也可以在虚拟机的 Ubuntu 上使用调试串口。首先要参考 11 手册安装好 Ubuntu20.04。在 Ubuntu上可以安装多种串口工具,比如 minicom,picocom,kermit。本章节我们来介绍 minicom 的安装和使用。 输入以下命令安装 minicomapt-get install…...
【http】HTTP/1.0、HTTP/1.1和HTTP/2.0
✨ 专栏介绍 在当今互联网时代,计算机网络已经成为了人们生活和工作中不可或缺的一部分。而要实现计算机之间的通信和数据传输,就需要依靠各种网络协议来进行规范和约束。无论是浏览网页、发送电子邮件还是进行在线交流,都离不开各种各样的网…...
automkcert使用教程
我们在开发的时候往往需要https开启一些浏览器功能,比如摄像头、wss等,自己的云服务器申请证书一个是麻烦,一个是价格贵,这种情况下可以用自签名证书。但自签名证书下有不通用的特点,其他机器无法快速信任自己的网站,因…...
tekton 发布 kubernetes 应用
tekton 发布 kubernetes 应用 基于Kubernetes 服务部署 Tekton Pipeline 实例,部署完成后使用tekton来完成源码拉取、应用打包、镜像推送和应用部署。 本文实现一个 golang-helloworld 项目 CI/CD 的完整流程,具体包括以下步骤: 从 gitee…...
unity脚本API中OnCollisionEnter()、OnTriggerEnter()二者的区别
Unity中的OnCollisionEnter和OnTriggerEnter两个函数在日常的开发中很常见但也容易混淆,下面说一说两者的区别。 碰撞器(Collider)与触发器(Trigger)的概念 碰撞器(Collider)和触发器ÿ…...
2023年12月【考试战报】|ORACLE OCP 19C考试通过
2023年10月【考试战报】|ORACLE OCP 19C考试通过-CSDN博客文章浏览阅读122次。自OCP认证进入中国以来,越来越被大多数DBA所认可,也越来越被企业所重视,90%以上DBA深造,都会选择OCP认证。随着OCP认证在全国范围内的普及,…...
鸿蒙操作系统:从手机到物联网,打造全场景智能体验
随着科技的不断发展,人们对于操作系统的需求也在不断升级。鸿蒙操作系统,作为华为推出的新一代智能终端操作系统,凭借其强大的分布式能力、流畅的用户体验以及丰富的应用生态,正逐渐成为人们关注的焦点。 一、鸿蒙操作系统概述 …...
[Ray Tracing: The Next Week] 笔记
前言 本篇博客参照自《Ray Tracing: The Next Week》教程,地址为:https://raytracing.github.io/books/RayTracingTheNextWeek.html 该教程在ray tracing in one weekend的基础上,增加了运动模糊、BVH树、Texture映射、柏林噪声、光照、体积…...
企业级实战项目:基于 pycaret 自动化预测公司是否破产
本文系数据挖掘实战系列文章,我跟大家分享一个数据挖掘实战,与以往的数据实战不同的是,用自动机器学习方法完成模型构建与调优部分工作,深入理解由此带来的便利与效果。 1. Introduction 本文是一篇数据挖掘实战案例,…...
dl转置卷积
转置卷积 转置卷积,顾名思义,通过名字我们应该就能看出来,其作用和卷积相反,它可以使得图像的像素增多 上图的意思是,输入是22的图像,卷积核为22的矩阵,然后变换成3*3的矩阵 代码如下 import…...
详解结构体(包含结构体内存对齐,柔性数组,位段)【尊嘟很详细】
结构体 结构体是一些值的集合,这些值称为成员变量,结构的成员可以是标量、数组、指针,甚至是其他结构体。 成员名可以与程序中其它变量同名,互不干扰。 结构体的定义 (struct结构名{}) struct books {int a;c…...
我的NPI项目之Android系统升级 - 同平台多产品的OTA
因为公司业务中涉及的面比较广泛,虽然都是提供移动终端PDA,但是使用的场景很多时候是不同的。例如,有提供给大型物流仓储的设备,对这样的设备必需具备扫码功能,键盘(戴手套操作),耐用…...
pnpm包管理器
官网 优点 快速 pnpm 比 npm 快了近 2 倍高效 node_modules 中的所有文件均克隆或硬链接自单一存储位置支持单体仓库 pnpm 内置了对单个源码仓库中包含多个软件包的支持权限严格 pnpm 创建的 node_modules 默认并非扁平结构,因此代码无法对任意软件包进行访问 安…...
flutter websocket发送ping包?
背景 服务端要求flutter客户端隔一段时间发送ping包,以此来建立心跳管理长连接。 代码 import package:web_socket_channel/io.dart; IOWebSocketChannel _channel IOWebSocketChannel.connect(Uri.parse(SocketService.url),pingInterval: const Duration(seco…...
基于采样的自动驾驶规划算法 - PRM,RRT,RRT*,CL-RRT
本文将讲解PRM,RRT,RRT*自动驾驶规划算法原理,不正之处望读者指正 0 前言 机器人运动规划的基本任务:从开始位置到目标位置的运动 (1)如何躲避构型空间出现的障碍物 (2)如何满足机器…...
CGAL的D维范围树和线段树
范围树和线段树是两种数据结构,用于高效地处理和查询数据。 范围树(Range Tree)是一种二叉树,它通过递归地将每个节点分割成两个子节点来存储一个点集。每个节点表示一个范围,并且存储该范围内所有点的最小和最大值。范…...
005.HCIA 传输层
传输层定义了主机应用程序之间端到端的连通性。传输层中最为常见的两个协议分别是传输控制协议TCP (Transmission Control Protocol)和用户数据包协议UDP (User Datagram Protocol)。 1、相关概念 a. 传输层的端口 端口范围:0-65535 知名端口:0-1023&…...
LLM之RAG实战(八)| 使用Neo4j和LlamaIndex实现多模态RAG
人工智能和大型语言模型领域正在迅速发展。一年前,没有人使用LLM来提高生产力。时至今日,很难想象我们大多数人或多或少都在使用LLM提供服务,从个人助手到文生图场景。由于大量的研究和兴趣,LLM每天都在变得越来越好、越来越聪明。…...
别再被CAPL路径搞懵了!getAbsFilePath、setFilePath这几个函数到底怎么用?
CAPL文件路径操作全解析:从函数原理到实战避坑指南 在CANoe自动化测试开发中,文件路径操作堪称最基础却又最容易出错的环节之一。许多工程师都经历过这样的场景:精心编写的CAPL脚本在本地测试一切正常,换到同事电脑上却频频报错&a…...
告别VirtualBox的‘不是Host-Only适配器’错误:一个网络配置的深度修复指南
VirtualBox Host-Only网络故障全解析:从原理到实战修复 当你正准备启动VirtualBox中的开发环境虚拟机时,突然弹出的红色错误提示框让所有工作戛然而止——"Interface is not a Host-Only Adapter"。这个看似简单的网络适配器错误背后…...
深入浅出DPCM与DAPM:图解高通音频架构如何实现动态功耗管理与低延迟播放
深入浅出DPCM与DAPM:图解高通音频架构如何实现动态功耗管理与低延迟播放 在智能穿戴设备和移动终端领域,音频系统的功耗优化一直是工程师面临的重大挑战。想象一下,当你的智能手表在待机状态下播放通知铃声时,如果每次都需要唤醒主…...
GEE实战:Landsat 8 TOA和SR数据去云处理,保姆级代码对比与避坑指南
GEE实战:Landsat 8 TOA与SR数据去云处理深度解析 当你在Google Earth Engine(GEE)平台上处理Landsat 8数据时,是否曾为选择TOA(大气层顶反射率)还是SR(地表反射率)而犹豫不决&#x…...
【技术解读】xNIDS:如何为深度学习入侵检测系统“翻译”可执行的主动防御规则?
1. 深度学习入侵检测的"黑盒困境":为什么需要翻译器? 第一次接触深度学习入侵检测系统(DL-NIDS)时,我被它的检测准确率惊艳到了——某些场景下能达到99%以上的识别率。但当我试图把它部署到实际生产环境时&a…...
CANN/asc-devkit DropOut高阶API
DropOut 【免费下载链接】asc-devkit 本项目是CANN 推出的昇腾AI处理器专用的算子程序开发语言,原生支持C和C标准规范,主要由类库和语言扩展层构成,提供多层级API,满足多维场景算子开发诉求。 项目地址: https://gitcode.com/ca…...
前端转行网络安全靠谱吗?靠挖漏洞变现可行吗
前言 最近,一个做运维的朋友跟我说他在学渗透测试。他说,公司请别人做渗透测试的费用是 2千/人天,一共2周。2周 2w 的收入,好香~ 于是,我也对渗透测试产生了兴趣。开始了探索之路~ 什么是渗透测试 渗透测试这名字听…...
告别命令行!用Offset Explorer(原Kafka Tool)图形化管理Kafka集群,5分钟上手
告别命令行恐惧:用Offset Explorer实现Kafka集群的可视化高效管理 对于许多开发者和运维人员来说,Kafka的命令行操作就像一道难以逾越的门槛。那些复杂的参数、冗长的命令和难以直观理解的输出,常常让人望而却步。而Offset Explorerÿ…...
从C++代码到机器指令:用OD和IDA手把手拆解一个简单的main函数(附寄存器图解)
从C代码到机器指令:用OD和IDA手把手拆解一个简单的main函数(附寄存器图解) 在逆向工程的世界里,理解高级语言如何转化为底层机器指令是一项基础而关键的技能。本文将以一个最简单的C main 函数为例,带你一步步追踪其从…...
3个步骤掌握LevelUI:可视化LevelDB数据库管理新体验
3个步骤掌握LevelUI:可视化LevelDB数据库管理新体验 【免费下载链接】levelui A GUI for LevelDB management based on atom-shell. 项目地址: https://gitcode.com/gh_mirrors/le/levelui 还在为LevelDB的命令行操作而烦恼吗?LevelUI为你带来了全…...

