当前位置: 首页 > article >正文

java基础 流(Stream)

Stream

  • Stream 的核心概念
    • 核心特点
  • Stream 的操作分类
    • 中间操作(Intermediate Operations)
    • 终止操作(Terminal Operations)
  • Stream 的流分类
    • 顺序流(Sequential Stream)
    • 并行流(Parallel Stream)
      • 并行流的注意事项
      • 并行流的底层机制
    • 顺序流 vs 并行流的对比
    • 顺序流和并行流的示例代码
    • 顺序流并行流总结

Stream 的核心概念

Java 8 引入的 Stream API 是一种基于函数式编程的数据处理抽象,允许以声明式方式操作集合(如过滤、映射、排序等)。Stream 不是数据结构,而是对数据源(集合、数组、I/O 等)的高效计算工具

核心特点

  • 链式调用:通过多个操作(中间操作 + 终止操作)串联处理数据
  • 惰性求值:中间操作(如 filter, map)不会立即执行,直到遇到终止操作(如 collect, forEach)
  • 不可复用:一个 Stream 只能被消费一次,再次使用会抛出 IllegalStateException
  • 并行处理:可通过简单方法(parallel())实现多线程并行计算

Stream 的操作分类

中间操作(Intermediate Operations)

返回新的 Stream,支持链式调用,常见方法示例:filter(), map(), sorted(), distinct()。

        List<Integer> list = Arrays.asList(1, 2, 2,3);list.stream().distinct().forEach(System.out::println);//用于去除流中的重复元素。list.stream().map(x -> x * x).forEach(System.out::println);//用于对流中的每个元素应用一个函数,并返回一个新的流list.stream().filter(v->v==3).forEach(System.out::println);//用于对流中的元素进行筛选,只保留满足条件的元素list.stream().flatMap(s -> s.toString().chars().boxed()).forEach(System.out::println);//用于将流中的每个元素转换为另一个流,然后将这些流连接成一个流list.stream().limit(2).forEach(System.out::println);//用于限制流中元素的数量list.stream().skip(2).forEach(System.out::println);//用于跳过流中前n个元素list.stream().sorted().forEach(System.out::println);//用于对流中的元素进行排序

支持链式调用

        list.stream().distinct().filter(v->v==3).sorted().map(x -> x * x).forEach(System.out::println);

但是注意,流中间操作的方法返回的结果依然是流,但是上面为了显示这些方法使用的样例里面
forEach(System.out::println)这个是终止操作方法,也就是在forEach方法之前的返回的是流,在使用forEach方法之后返回的又将流转换非 Stream 结果

终止操作(Terminal Operations)

触发实际计算,返回非 Stream 结果(如集合、数值、void),常见方法示示例:collect(), forEach(), reduce(), count()

        List<String> names = Arrays.asList("Alice", "Bob", "Charlie");names.stream().forEach(System.out::println);//对流中的每个元素执行指定的操作 这里是打印names.stream() .map(String::toUpperCase).collect(Collectors.toList()).forEach(System.out::println);//将流中的元素收集到一个容器中System.out.println(names.stream().allMatch(s -> s.contains("s")));//allMatch 检查是否匹配所有元素System.out.println(names.stream().anyMatch(s -> s.contains("a")));//anyMatch 检查是否至少匹配一个元素System.out.println(names.stream().noneMatch(s -> s.contains("x")));//noneMatch:检查是否没有匹配所有元素System.out.println(names.stream().findFirst());//findFirst:返回当前流中的第一个元素System.out.println(names.stream().findAny());//findAny:返回当前流中的任意元素System.out.println(names.stream().count());//count:返回流中元素总数System.out.println(names.stream().max(Comparator.comparingInt(s->s.length())));//max:返回流中最大值System.out.println(names.stream().min(Comparator.comparingInt(s->s.length())));//min:最小值

Stream 的流分类

顺序流(Sequential Stream)

默认模式:所有操作在单线程中按顺序执行

List<Integer> list = Arrays.asList(1, 2, 3);
Stream<Integer> stream = list.stream(); // 顺序流

适用场景:
数据量较小,或操作本身简单。
需要保证操作顺序(如 sorted() 依赖前序操作结果)。

并行流(Parallel Stream)

Stream<Integer> parallelStream = list.parallelStream(); // 并行流Stream<Integer> parallelStream = list.stream().parallel(); // 转换为并行流

适用场景:
数据量较大,且任务可独立拆分(无共享状态或顺序依赖)。
操作耗时(如复杂计算、I/O 等待)。

并行流的注意事项

1.线程安全问题
确保操作中使用的 Lambda 表达式或函数是线程安全的(避免共享可变状态)。
示例错误:

List<Integer> result = new ArrayList<>();
list.parallelStream().forEach(result::add); // 并发修改 ArrayList 导致数据错误

应使用线程安全的收集器(如 Collectors.toList()):

List<Integer> result = list.parallelStream().collect(Collectors.toList());

2.性能未必更好
并行流需额外开销(任务拆分、线程调度),对小数据量可能更慢。
测试性能:通过基准测试(如 JMH) 验证是否适合并行

3.顺序依赖操作
如 limit()、findFirst() 在并行流中可能性能更差,需改用无序流

list.parallelStream().unordered().limit(10); // 提升性能

ORDERED是Spliterator的特征值, 特征要求并行流保持元素顺序(如 List),改成unordered就是改成序列流,除此之外

并行流的底层机制

1.Spliterator
并行流通过 Spliterator(可拆分迭代器) 将数据源拆分为多个子任务,是 Stream API 并行处理的底层实现

List<String> list = Arrays.asList("Java", "Python", "C++", "Go");// 使用并行流处理
list.parallelStream() // 将数据源拆分为多个子任务.map(String::toUpperCase).forEach(System.out::println);// 输出(顺序不确定,因为并行处理):
// PYTHON
// JAVA
// GO
// C++

2.ForkJoinPool
并行流默认使用公共的 ForkJoinPool(线程数 = CPU 核心数)。通过系统属性修改默认线程池大小

System.setProperty("java.util.concurrent.ForkJoinPool.common.parallelism", "8");

3.自定义线程池(避免影响全局)

ForkJoinPool customPool = new ForkJoinPool(4);
customPool.submit(() -> list.parallelStream().forEach(...)).get();

顺序流 vs 并行流的对比

特性顺序流并行流
线程模型单线程多线程(ForkJoinPool 默认线程池)
性能优势简单任务、小数据量大数据量、可并行化的复杂任务
开销高(线程切换、任务拆分/合并)
顺序保证严格按顺序处理不保证顺序(除非使用有序操作)
数据源要求无特殊要求数据源需可拆分(如 ArrayList

顺序流和并行流的示例代码

顺序流处理

List<String> names = Arrays.asList("Alice", "Bob", "Charlie");
List<String> filtered = names.stream().filter(name -> name.length() > 3).map(String::toUpperCase).collect(Collectors.toList()); // [ALICE, CHARLIE]

并行流处理

List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
int sum = numbers.parallelStream()//通过底层Spliterator拆分为多个子任务.filter(n -> n % 2 == 0).mapToInt(n -> n * 2).sum(); // (2+4+6+8+10)*2 = 60

顺序流并行流总结

顺序流:简单、低开销,适合小数据量或顺序敏感操作。
并行流:通过多线程加速处理,适合大数据量和可并行化任务,但需注意线程安全和性能开销。

选择策略:
优先使用顺序流,仅在必要时(且验证有效)切换为并行流。
避免在并行流中操作共享可变状态。

相关文章:

java基础 流(Stream)

Stream Stream 的核心概念核心特点 Stream 的操作分类中间操作&#xff08;Intermediate Operations&#xff09;终止操作&#xff08;Terminal Operations&#xff09; Stream 的流分类顺序流&#xff08;Sequential Stream&#xff09;并行流&#xff08;Parallel Stream&…...

【AI】prompt engineering

prompt engineering ## prompt engineering ## prompt engineering ## prompt engineering 一、定义 Prompt 工程&#xff08;Prompt Engineering&#xff09;是指在使用语言模型&#xff08;如 ChatGPT、文心一言等&#xff09;等人工智能工具时&#xff0c;设计和优化输入提…...

无需libpacp库,BPF指令高效捕获指定数据包

【环境】无libpacp库的Linux服务器 【要求】高效率读取数据包&#xff0c;并过滤指定端口和ip 目前遇到两个问题 一是手写BPF&#xff0c;难以兼容&#xff0c;有些无法正常过滤二是性能消耗问题&#xff0c;尽可能控制到1% 大方向&#xff1a;过滤数据包要在内核层处理&…...

LeetCode算法题(Go语言实现)_36

题目 给定一个二叉树的根节点 root &#xff0c;和一个整数 targetSum &#xff0c;求该二叉树里节点值之和等于 targetSum 的 路径 的数目。 路径 不需要从根节点开始&#xff0c;也不需要在叶子节点结束&#xff0c;但是路径方向必须是向下的&#xff08;只能从父节点到子节点…...

react实现上传图片到阿里云OSS以及问题解决(保姆级)

一、优势 提高上传速度&#xff1a;前端直传利用了浏览器与 OSS 之间的直接连接&#xff0c;能够充分利用用户的网络带宽。相比之下&#xff0c;后端传递文件时&#xff0c;文件需要经过后端服务器的中转&#xff0c;可能会受到后端服务器网络环境和处理能力的限制&#xff0c;…...

无法看到新安装的 JDK 17

在 Linux 系统中使用 update-alternatives --config java 无法看到新安装的 JDK 17&#xff0c;可能是由于 JDK 未正确注册到系统备选列表中。 一、原因分析 JDK 未注册到 update-alternatives update-alternatives 工具需要手动注册 JDK 路径后才能识别新版本。如果仅安装 JDK…...

LeetCode 3396.使数组元素互不相同所需的最少操作次数:O(n)一次倒序遍历

【LetMeFly】3396.使数组元素互不相同所需的最少操作次数&#xff1a;O(n)一次倒序遍历 力扣题目链接&#xff1a;https://leetcode.cn/problems/minimum-number-of-operations-to-make-elements-in-array-distinct/ 给你一个整数数组 nums&#xff0c;你需要确保数组中的元素…...

Vue2 快速过度 Vue3 教程 (后端学习)

隔好长一段时间没有写文章了&#xff0c;因为最近公司一个项目进度很赶&#xff0c;导致一直加班&#xff0c;没有时间空出来学习新的东西&#xff0c;这次趁着周末&#xff0c;赶紧补一下之前落下的一直想重新学一下整个大前端生态的想法&#xff0c;这次写一篇自己学习Vue3的…...

供应链管理-职业规划:数字化供应链管理专家 / 供应链管理商业模式专家 / 供应链管理方案专家

一、背景阐述 依据联合国产业分类标准&#xff0c;工业体系被细致划分为41个工业大类、207个工业中类以及666个工业小类。中国凭借其独特的产业布局&#xff0c;成为全球唯一一个全面涵盖所有这些门类的国家&#xff0c;成功构建起独立且完备的现代工业体系。这一辉煌成就&…...

无状态版的DHCPv6是不是SLAAC? 笔记250405

无状态版的DHCPv6是不是SLAAC? 笔记250405 无状态版 DHCPv6 不是 SLAAC&#xff0c;但二者在 IPv6 网络中可协同工作。以下是核心区别与协作关系&#xff1a; 本质区别 特性SLAAC无状态 DHCPv6主要功能生成 IPv6 地址&#xff08;基于路由器通告的前缀&#xff09;分发 DNS、…...

遍历算法及其应用详解

李升伟 整理 什么是遍历&#xff1f; 遍历是指按照某种规则或顺序&#xff0c;系统地访问数据结构&#xff08;如树、图等&#xff09;中的每个节点一次且仅一次的过程。遍历是算法设计中的基本操作&#xff0c;用于访问、检查或修改数据结构中的所有元素。 主要遍历算法 1…...

Python 字典和集合(常见的映射方法)

本章内容的大纲如下&#xff1a; 常见的字典方法 如何处理查找不到的键 标准库中 dict 类型的变种set 和 frozenset 类型 散列表的工作原理 散列表带来的潜在影响&#xff08;什么样的数据类型可作为键、不可预知的 顺序&#xff0c;等等&#xff09; 常见的映射方法 映射类型…...

基于大模型的ALS预测与手术优化系统技术方案

目录 技术方案文档:基于大模型的ALS预测与手术优化系统1. 数据预处理与特征工程模块流程图伪代码2. 多模态融合预测模型模型架构图伪代码3. 术中实时监测与动态干预系统系统流程图伪代码4. 统计验证与可解释性模块验证流程图伪代码示例(SHAP分析)5. 健康教育与交互系统系统架…...

创建一个简单的HTML游戏站

创建一个简单的HTML游戏站涉及多个步骤&#xff0c;包括规划网站结构、设计用户界面、编写游戏逻辑以及测试和部署。下面是一个详细的步骤指南&#xff1a; 1. 规划网站结构 确定目标受众&#xff1a;了解你的目标用户群体。选择游戏类型&#xff1a;决定你要开发的游戏类型&…...

Matlab轴承故障信号仿真与故障分析

1.摘要 本文介绍了一个基于Matlab的轴承故障信号仿真与分析程序&#xff0c;旨在模拟和分析轴承内圈故障信号的特征。程序首先通过生成故障信号、共振信号和调制信号&#xff0c;添加噪声和离散化处理&#xff0c;构建模拟的振动信号&#xff0c;并保存相关数据。通过快速傅里…...

Linux 进程 | 概念 / 特征 / 状态 / 优先级 / 空间

注&#xff1a; 本文为 “Linux 进程” 相关文章合辑。 未整理去重。 Linux 进程概念&#xff08;精讲&#xff09; A little strawberry 于 2021-10-15 10:23:55 发布 基本概念 课本概念&#xff1a;程序的一个执行实例&#xff0c;正在执行的程序等。 内核观点&#xff…...

项目中如何防止超卖

什么是超卖&#xff1f;假如只剩下一个库存&#xff0c;却被多个订单买到了&#xff0c;简单理解就是库存不够了还能正常下单。 方案1&#xff1a;数据库行级锁 1. 实体类 Data TableName("product") public class Product {TableId(type IdType.AUTO)private Lon…...

重回全面发展亲自操刀

项目场景&#xff1a; 今年工作变动&#xff0c;优化后在一家做国有项目的私人公司安顿下来了。公司环境不如以前&#xff0c;但是好在瑞欣依然可以每天方便的买到。人文氛围挺好&#xff0c;就是工时感觉有点紧&#xff0c;可能长期从事产品迭代开发&#xff0c;一下子转变做项…...

3D珠宝渲染用什么软件比较好?渲染100邀请码1a12

印度珠宝商 Mohar Fine Jewels 和英国宝石商 Gemfields 在今年推出了合作珠宝系列——「Emeralds in Full Bloom」&#xff0c;它的灵感源自花草绽放的春季田野&#xff0c;共有 39 件作品&#xff0c;下面这个以植物为主题的开口手镯就是其中一件。 在数字时代&#xff0c;像这…...

【数据结构】邻接矩阵完全指南:原理、实现与稠密图优化技巧​

邻接矩阵 导读一、图的存储结构1.1 分类 二、邻接矩阵法2.1 邻接矩阵2.2 邻接矩阵存储网 三、邻接矩阵的存储结构四、算法评价4.1 时间复杂度4.2 空间复杂度 五、邻接矩阵的特点5.1 特点1解析5.2 特点2解析5.3 特点3解析5.4 特点4解析5.5 特点5解析5.6 特点6解析 结语 导读 大…...

【嵌入式-stm32电位器控制以及旋转编码器控制LED亮暗】

嵌入式-stm32电位器控制LED亮暗 任务1代码1Key.cKey.hTimer.cTimer.hPWM.cPWM.hmain.c 实验现象1任务2代码2Key.cKey.hmain.c 实验现象2问题与解决总结 源码框架取自江协科技&#xff0c;在此基础上做扩展开发。 任务1 本文主要介绍利用stm32f103C8T6实现电位器控制PWM的占空比…...

ragflow开启https访问:添加证书后,使用浏览器还是有警告,如何解决?

如果在 Windows 系统中安装了 PEM 证书(使用方法一通过证书管理器 MMC 导入),但浏览器仍然提示安全警告,可能有以下几个原因及解决方法: 1. 证书未正确安装到受信任的存储位置 问题:如果证书被导入到错误的存储位置(如“个人”而非“受信任的根证书颁发机构”),浏览器…...

字符串——面试考察高频算法题

目录 转换成小写字母 字符串转化为整数 反转相关的问题 反转字符串 k个一组反转 仅仅反转字母 反转字符串里的单词 验证回文串 判断是否互为字符重排 最长公共前缀 字符串压缩问题 转换成小写字母 给你一个字符串 s &#xff0c;将该字符串中的大写字母转换成相同的…...

Uniapp 集成极光推送(JPush)完整指南

文章目录 前言一、准备工作1. 注册极光开发者账号2. 创建应用3. Uniapp项目准备 二、集成极光推送插件方法一&#xff1a;使用UniPush&#xff08;推荐&#xff09;方法二&#xff1a;手动集成极光推送SDK 三、配置原生平台参数四、核心功能实现1. 获取RegistrationID2. 设置别…...

Plusar集群搭建-Ubuntu20.04-Winterm

1 背景 已经部署了Pulsar集群在生产上&#xff0c;新项目需要用到Pulsar。对Pulsar不熟&#xff0c;故搭建练手。 环境&#xff1a;Windows10vmwareUbuntu20.04&#xff0c;ssh工具使用的Winterm。 使用的是root账户&#xff0c;ubuntu防火墙都ufw disable了。 2 参考文档 集…...

selenium元素获取

from selenium import webdriver from selenium.webdriver.common.by import Bydriver webdriver.Chrome()driver.maximize_window()#最大化窗口 #隐式等待 driver.implicitly_wait(10)#打开网页 driver.get("https://www.zhipin.com/beijing/?kacity-sites-101010100&q…...

AI比人脑更强,因为被植入思维模型【50】邓克效应思维模型

giszz的理解&#xff1a;DK Effect&#xff0c;就是井底之蛙。这里有个启发&#xff0c;就是人的认知提升&#xff0c;有4个阶段&#xff0c;愚昧区、崩溃区、成长区、智慧区。也分别对应4个境界&#xff1a;自然境界、功利境界、道德境界、天地境界。我个人觉得自己刚刚过了崩…...

8、nRF52xx蓝牙学习(boards.h文件学习)

boards.h文件的代码如下&#xff1a; #ifndef BOARDS_H #define BOARDS_H#include "nrf_gpio.h" #include "nordic_common.h"#if defined(BOARD_NRF6310)#include "nrf6310.h" #elif defined(BOARD_PCA10000)#include "pca10000.h" #…...

声明文件.d.ts

在 TypeScript 中&#xff0c;.d.ts 文件是类型声明文件&#xff08;Declaration Files&#xff09;&#xff0c;用于描述 JavaScript 库或模块的类型信息&#xff0c;但不包含具体实现。它们帮助 TypeScript 编译器进行类型检查&#xff0c;同时保持与纯 JavaScript 的兼容性。…...

java整合socket通信全流程

前言 大家好,由于工作上业务的需要,在java项目中引入了socket通信,特此记录一下,用以备份,本文章中的socket通信实现了,服务端与客户端的双向通讯,以及二者之间的心跳通信,服务端重启之后,客户端的自动重连功能。 原理 Socket通信是计算机网络中常用的一种通信机制…...