Java流式处理-Steam详解
Java 8 引入的 Stream API 是一种强大的处理集合数据的工具,它允许你以声明式方式处理数据集合(如 List、Set 等),并支持多种聚合操作(如过滤、映射、排序、归约等)。Stream API 可以显著提高代码的可读性和简洁性,同时支持并行处理以提升性能。
一、Stream 基础概念
1. 什么是 Stream?
- Stream 是一种针对集合数据的高级抽象,它不存储数据,而是定义了一系列对数据源(如集合、数组)的操作。
- Stream 操作是惰性的,只有在终止操作被调用时才会执行。
- Stream 可以是串行的或并行的,并行流利用多线程提高处理大数据的效率。
2. Stream 的特点
- 不改变数据源:Stream 操作不会修改原始数据,而是返回一个新的 Stream。
- 一次性使用:Stream 只能被消费一次,消费后需要重新创建。
- 延迟执行:中间操作(如 filter、map)只会记录操作,不会立即执行,直到终止操作(如 collect、forEach)被调用。
二、Stream 的创建
1. 从集合创建
java
List<String> list = Arrays.asList("apple", "banana", "cherry");
Stream<String> stream = list.stream(); // 串行流
Stream<String> parallelStream = list.parallelStream(); // 并行流
2. 从数组创建
java
int[] array = {1, 2, 3, 4, 5};
IntStream intStream = Arrays.stream(array);
3. 使用 Stream.of ()
java
Stream<String> stream = Stream.of("apple", "banana", "cherry");
4. 创建无限流
java
// 生成无限序列(0, 2, 4, 6...)
Stream<Integer> evenNumbers = Stream.iterate(0, n -> n + 2);
// 生成随机数流
Stream<Double> randoms = Stream.generate(Math::random);
三、Stream 的中间操作
中间操作返回一个新的 Stream,允许链式调用多个操作。常见的中间操作包括:
1. 过滤元素:filter()
java
List<String> fruits = Arrays.asList("apple", "banana", "cherry", "date");
List<String> filtered = fruits.stream().filter(fruit -> fruit.startsWith("a")) // 保留以 'a' 开头的元素.collect(Collectors.toList()); // [apple]
2. 映射元素:map()
java
List<Integer> numbers = Arrays.asList(1, 2, 3);
List<Integer> squared = numbers.stream().map(n -> n * n) // 平方每个元素.collect(Collectors.toList()); // [1, 4, 9]
3. 扁平化嵌套结构:flatMap()
java
List<List<Integer>> nested = Arrays.asList(Arrays.asList(1, 2),Arrays.asList(3, 4)
);
List<Integer> flattened = nested.stream().flatMap(List::stream) // 将嵌套列表展开为单个流.collect(Collectors.toList()); // [1, 2, 3, 4]
4. 排序:sorted()
java
List<String> names = Arrays.asList("Charlie", "Alice", "Bob");
List<String> sorted = names.stream().sorted() // 自然排序.collect(Collectors.toList()); // [Alice, Bob, Charlie]
5. 去重:distinct()
java
List<Integer> duplicates = Arrays.asList(1, 2, 2, 3, 3, 3);
List<Integer> unique = duplicates.stream().distinct() // 去重.collect(Collectors.toList()); // [1, 2, 3]
6. 截断:limit()
和 skip()
java
List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5);
List<Integer> limited = numbers.stream().limit(3) // 取前3个元素.collect(Collectors.toList()); // [1, 2, 3]List<Integer> skipped = numbers.stream().skip(2) // 跳过前2个元素.collect(Collectors.toList()); // [3, 4, 5]
四、Stream 的终止操作
终止操作触发 Stream 的执行并返回最终结果。常见的终止操作包括:
1. 收集结果:collect()
java
// 收集到 List
List<String> resultList = stream.collect(Collectors.toList());// 收集到 Set
Set<String> resultSet = stream.collect(Collectors.toSet());// 收集到 Map
Map<Integer, String> map = stream.collect(Collectors.toMap(s -> s.length(), // key: 字符串长度s -> s, // value: 字符串本身(existing, replacement) -> existing // 处理键冲突的策略));
2. 遍历元素:forEach()
java
stream.forEach(System.out::println); // 打印每个元素
3. 聚合操作:reduce()
java
// 求和
int sum = numbers.stream().reduce(0, (a, b) -> a + b); // 初始值为0,累加所有元素// 求最大值
Optional<Integer> max = numbers.stream().reduce(Integer::max); // 返回 Optional 避免空指针
4. 匹配元素:anyMatch()
、allMatch()
、noneMatch()
java
boolean anyStartsWithA = stream.anyMatch(s -> s.startsWith("A")); // 是否存在以 'A' 开头的元素
boolean allStartsWithA = stream.allMatch(s -> s.startsWith("A")); // 是否所有元素都以 'A' 开头
boolean noneStartsWithA = stream.noneMatch(s -> s.startsWith("A")); // 是否没有元素以 'A' 开头
5. 查找元素:findFirst()
、findAny()
java
Optional<String> first = stream.findFirst(); // 返回第一个元素(可能为空)
Optional<String> any = stream.findAny(); // 返回任意元素(并行流中可能随机返回)
6. 统计:count()
、min()
、max()
java
long count = stream.count(); // 元素个数
Optional<Integer> min = numbers.stream().min(Integer::compareTo); // 最小值
Optional<Integer> max = numbers.stream().max(Integer::compareTo); // 最大值
五、并行流(Parallel Stream)
并行流利用多核处理器并行执行 Stream 操作,适用于大数据集的高效处理。
1. 创建并行流
java
List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5);
Stream<Integer> parallelStream = numbers.parallelStream(); // 直接创建并行流
Stream<Integer> parallelStream2 = numbers.stream().parallel(); // 将串行流转换为并行流
2. 并行流示例
java
long sum = numbers.parallelStream().mapToInt(Integer::intValue).sum(); // 并行求和
3. 注意事项
- 线程安全:并行流操作涉及多线程,确保操作的数据源和中间操作是线程安全的。
- 适用场景:对于小数据集或操作本身开销不大的情况,并行流可能比串行流更慢(线程调度开销)。
- 顺序问题:并行流不保证处理顺序,若需要顺序性,可使用
forEachOrdered()
替代forEach()
。
六、Collectors 工具类
Collectors
是一个实用工具类,提供了各种收集器(Collector),用于将 Stream 结果收集到不同的数据结构中。
1. 分组:groupingBy()
java
// 按字符串长度分组
Map<Integer, List<String>> groups = stream.collect(Collectors.groupingBy(String::length));
2. 分区:partitioningBy()
java
// 按条件分区(偶数和奇数)
Map<Boolean, List<Integer>> partitioned = numbers.stream().collect(Collectors.partitioningBy(n -> n % 2 == 0));
3. 连接字符串:joining()
java
String joined = stream.collect(Collectors.joining(", ")); // 用逗号连接元素
4. 统计:summarizingInt()
java
IntSummaryStatistics stats = numbers.stream().collect(Collectors.summarizingInt(Integer::intValue));
// stats 包含 count, sum, min, max, average
七、Stream 与 Lambda 表达式结合
Stream API 与 Lambda 表达式紧密结合,使代码更简洁。例如:
java
// 传统方式
List<String> filtered = new ArrayList<>();
for (String fruit : fruits) {if (fruit.startsWith("a")) {filtered.add(fruit);}
}// Stream + Lambda 方式
List<String> filtered = fruits.stream().filter(fruit -> fruit.startsWith("a")).collect(Collectors.toList());
八、常见使用场景
1. 数据过滤与转换
java
List<Product> products = getProducts();
List<String> productNames = products.stream().filter(p -> p.getPrice() > 100) // 过滤价格大于100的产品.map(Product::getName) // 提取产品名称.collect(Collectors.toList());
2. 聚合统计
java
double averagePrice = products.stream().mapToDouble(Product::getPrice).average() // 计算平均价格.orElse(0.0);
3. 并行处理
java
// 并行计算所有产品的总价格
double totalPrice = products.parallelStream().mapToDouble(Product::getPrice).sum();
总结
Java Stream API 通过声明式方式处理集合数据,使代码更简洁、易读,并支持并行处理以提高性能。关键要点:
- 中间操作(如
filter
、map
)返回新的 Stream,支持链式调用。 - 终止操作(如
collect
、forEach
)触发执行并产生结果。 - 并行流适用于大数据集的高效处理,但需注意线程安全和适用场景。
- Collectors 工具类提供丰富的收集器,用于结果转换和聚合。
相关文章:
Java流式处理-Steam详解
Java 8 引入的 Stream API 是一种强大的处理集合数据的工具,它允许你以声明式方式处理数据集合(如 List、Set 等),并支持多种聚合操作(如过滤、映射、排序、归约等)。Stream API 可以显著提高代码的可读性和…...

windows服务器部署jenkins工具(一)
jenkins作为一款常用的构建发布工具,极大的简化了项目部署发布流程。jenkins通常是部署在linux服务上,今天给大家分享的是windows服务器上如何搭建jenkins发布工具。 1.首先第一步还是看windows安装docker 这篇文章哈,当然也可以不采用docke…...
LCS4110R加密芯片在打印机墨盒的应用
在打印机耗材行业,始终有一部分用户在谋求以某种方式破解、绕开厂商采取的各种限制措施使用第三方墨盒,低价克隆墨盒泛滥导致原厂利润流失、用户体验下降,甚至引发设备损坏风险。所以墨盒的兼容性与安全性一直是品牌商与用户的共同痛点。如何…...
什么是 API 管理?为什么管理 API 很重要?如何用 iPaaS 平台管理 API
在当今数字化浪潮下,API(ApplicationProgrammingInterface,应用程序编程接口)作为连接不同系统、应用程序与设备的关键桥梁,其重要性日益凸显。而API管理则是把控API全生命周期,使其稳定、可靠、安全运转并…...

基于51单片机和8X8点阵屏、独立按键的飞行躲闪类小游戏
目录 系列文章目录前言一、效果展示二、原理分析三、各模块代码1、8X8点阵屏2、独立按键3、定时器04、定时器1 四、主函数总结 系列文章目录 前言 用的是普中A2开发板。 【单片机】STC89C52RC 【频率】12T11.0592MHz 【外设】8X8点阵屏、独立按键 效果查看/操作演示ÿ…...

告别“盘丝洞”车间:4-20mA无线传输如何重构工厂神经网?
4-20ma无线传输是利用无线模块将传统的温度、压力、液位等4-20mA电流信号转换为无线信号进行传输。这一技术突破了有线传输的限制,使得信号可以在更广泛的范围内进行灵活、快速的传递,无线传输距离可达到50KM。达泰4-20ma无线传输模块在实现工业现场应用…...

VMware虚拟机突然无法ssh连接
遇到的情况: 功能全部正常的情况下,没有修改任何配置,重启电脑之后无法ssh连接 其实不太可能的可能原因: 1、虚拟机内部sshd服务未运行 systemctl status sshd systemctl start sshd 2、检查SSH端口监听 netstat -an | grep :…...
Android帧抢占协议技术剖析:触摸事件与UI绘制的智能调度优化方案
简介 在移动应用开发中,触摸事件响应与UI绘制的同步竞争是导致卡顿和掉帧的主要原因之一。腾讯工程师提出的优先级策略通过紧急事件抢占、增量渲染机制和时间片补偿技术,有效解决了这一竞争问题。本文将深入分析这些技术原理,并提供完整的代码实现,帮助开发者构建更流畅的…...
Maven 项目介绍
一、Maven 概述 Maven 是一个基于 Java 的项目管理和构建自动化工具,由 Apache 软件基金会开发。它采用 “约定优于配置”(Convention Over Configuration)的原则,通过标准化的项目结构和配置,极大地简化了项目的构建…...

班迪录屏--解决视频剪辑时声音和画面不同步的问题
原文网址:班迪录屏--解决视频剪辑时声音和画面不同步的问题_IT利刃出鞘的博客-CSDN博客 简介 本文介绍如何用班迪录屏解决视频剪辑时声音和画面不同步的问题。 问题描述 我用班迪录屏录了视频,用剪映进行剪辑,结果发现在剪辑时声音和画面…...

Git上传项目到GitHub
Git上传项目到GitHub 下载Git客户端配置Git设置GitHub上传本地项目到Github 下载Git客户端 网址:Git Windows客户端。选择Standalone Installer(单独安装程序),并点击64bit Git for Windows Setup(64位Git for Windows安装程序)进行下载。然后一路默认选…...

【工具】Quicker/VBA|PPT 在指定位置添加有颜色的参考线
文章目录 效果展示使用方式技术原理更多原理ActivePresentation.Guides 概述主要属性和方法使用示例添加水平参考线添加垂直参考线删除所有参考线获取参考线数量 注意事项 致谢 效果展示 先展示效果: Quicker 动作:VBA 添加参考线 - Quicker 动作 使用…...

第34节:迁移学习中的特征提取方法
迁移学习中的特征提取方法:原理、技术与应用 1. 迁移学习与特征提取概述 迁移学习(Transfer Learning)作为机器学习领域的重要范式 通过将源领域(source domain)学到的知识迁移到目标领域(target domain),有效解决了传统机器学习需要大量标注数据的瓶颈问题。 在迁…...
C语言中的文件I/O
C标准I/O库函数 1、C标准I/O库函数1.1、打开/写入/读取/关闭文件1.2、读取文件数据1.2.1、fgetc1.2.2、fgets1.2.3、fscanf1.3、标准输入/输出/错误2、系统调用2.1、常用的系统调用2.2、综合小demo3、文件描述符3.1、定义3.2、`文件描述符`关联的数据结构3.2.1、struct file3.2…...

(万字长文)Django数据库操作——ORM:数据交互显示前端网页
🌟 如果这篇文章触动了你的心弦,请不要吝啬你的支持! 亲爱的读者, 感谢你花时间阅读这篇分享。希望这里的每一个字都能为你带来启发或是让你会心一笑。如果你觉得这篇文章有价值,或者它解决了你一直以来的一个疑问&a…...

实验-使用递归计算阶乘-RISC-V(计算机组成原理)
目录 一、实验内容 二、实验步骤 三、实验效果 四、实验环境 五、实验小结和思考 一、实验内容 一个典型的计算阶乘的递归过程如下图所示: 在这个任务中,一份汇编代码的框架“task4-阶乘”你需要使用RISC-V或MIPS汇编程序以递归的形式解决这个问题。…...

ISO 26262-5 评估硬件架构度量值
两种硬件架构的度量, 用于评估相关项架构应对随机硬件失效的有效性。 应评估(评估仅限于ASIL (B)、 C 和 D 的安全目标) 1 应将按照附录 C 单点故障度量和潜伏故障度量的诊断覆盖率来评估 2 应结合残余故障和相关的潜伏故障来预估安全机制…...
JMeter 教程:响应断言
目录 JMeter 教程:响应断言的简单介绍【轻松上手】 ✅ 什么是响应断言? 📌 使用场景示例 🛠️ 添加响应断言步骤 1. 选中 HTTP 请求 → 右键 → Add → Assertions → Response Assertion 2. 设置断言内容: ✅ …...

【Qt开发】显示类控件——QLCDNumber
目录 1,QLCDNumber的说明 2,QLCDNumber的运用 1,QLCDNumber的说明 QLCDNumer 是一个专门用来显示数字的控件。它类似于 "老式计算器" 的效果。它的核心属性如下: 2,QLCDNumber的运用 定时器 运用QLCDNumb…...
深入剖析 5G 核心网中的 PLMN
一、引言 在 5G 技术迅猛发展的当下,5G 核心网作为整个通信系统的关键枢纽,支撑着海量数据传输、低延迟通信以及多样化业务应用。其中,公共陆地移动网络(Public Land Mobile Network,PLMN)扮演着极为重要的角色,它是 5G 核心网实现用户接入、网络管理以及业务提供的基础…...

音频AAC编码与RV1126的AENC模块的讲解
一.音频编码的原理 AAC编码的基本概念 AAC(Advanced Audio Coding)是一种高级音频编码格式,旨在提供比MP3更高的音质和更低的比特率。AAC是MPEG-2和MPEG-4标准的一部分,广泛应用于音乐、视频流媒体和广播等领域 音频为什么要进…...

vue页面目录菜单有些属性是根据缓存读取的。如果缓存更新了。希望这个菜单也跟着更新。
父组件中有两个子组件。如果在B组件数据更新之后。A组件也跟着一起改变呢?如图如果我右边基本信息里面勾选了高血压,左侧菜单里面也要立刻出现一个高血压随访菜单,如果我取消勾选了左侧菜单就去掉。 左侧菜单的显示和隐藏的数据实际上是放在…...
Android开发-Application
在Android应用开发中,Application类扮演着非常重要的角色。它作为整个应用程序的全局单例实例存在,在应用启动时最先被创建,并且在整个应用生命周期内持续存在。通过自定义Application类,开发者可以执行全局初始化操作、管理全局状态或数据等。本文将详细介绍Application的…...

在TIA 博途中下载程序时找不到对应的网卡怎么办?
1. 检查物理连接 确认网线已正确连接PLC和PC,接口指示灯正常。 尝试更换网线或交换机端口,排除硬件故障。 2. 确认网卡驱动已安装 设备管理器检查: 右键点击“此电脑” → “管理” → “设备管理器”。 展开“网络适配器”,确…...

《量子计算实战》PDF下载
内容简介 在加密、科学建模、制造物流、金融建模和人工智能等领域,量子计算可以极大提升解决问题的效率。量子系统正变得越来越强大,逐渐可用于生产环境。本书介绍了量子计算的思路与应用,在简要说明与量子相关的科学原理之后,指…...

Linux入门(部分基础相关知识+常用命令+权限)
目录 1.基础背景了解 2、基本操作系统、linux相关知识 1.操作系统是一款用来管理软硬件资源的软件。 2.对于一个文件来说,是由文件内容文件属性构成的。空文件(内容为空)也占磁盘空间。 3.linux下的目录结构 4.linux下的删除 5.环境 6…...

海拔案例分享-实践活动报名测评小程序
大家好,今天湖南海拔科技想和大家分享一款实践活动报名测评小程序,客户是长沙一家专注青少年科创教育的机构,这家机构平时要组织各种科创比赛、培训课程,随着学员增多,管理上的问题日益凸显:每次组织活动&a…...
JavaWeb面试题 (一)
1. 常见的软件系统结构 1. C/S C/S结构即客户端/服务器(Client/Server),例如QQ; 缺点:软件更新时需要同时更新客户端和服务器端两端,比较麻烦; 优点:安全性比较好 2. B/S B/S结…...
解决Vue项目依赖错误:使用electron-vite重建
文章目录 开端解决方案:使用 electron-vite Vue 重建项目1. 环境准备2. 创建新项目3. 安装依赖并启动项目 开端 在开发过程中,我遇到了一个令人头疼的错误提示: 0:0 error Parsing error: Cannot find module vue/cli-plugin-babel/preset…...
vue3样式穿透用法
在Vue3中,样式穿透可通过以下方式实现: 1. 基础用法 使用::v-deep伪类实现样式穿透(兼容Vue2语法): .parent ::v-deep .child-component {color: red; }2. 推荐方式 使用:deep()伪函数(Vue3推荐写法&am…...