news 2026/2/25 21:48:10

Java Stream API 使用总结

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Java Stream API 使用总结

Java 8之后,Stream API 已经成为日常开发中不可或缺的工具
但很多人对 Stream 的理解,仍停留在 filter + map + collect。
本文结合真实业务场景,总结 Stream API 的正确使用方式、常见误区和进阶技巧


一、为什么要用 Stream API

直奔主题嗷 ~ ,我们首先结合一组代码来体会一下不用 Stream API 和使用 Stream API 带给我们的便捷。

下图左方代码是没有使用 Stream API 的代码,我们通过增强 for 循环对 List 中的元素进行过滤,分离出其中年龄 > 18 的用户信息;右图则是使用了 Stream API 的代码,看出来了吗,代码的简洁程度、可读性?即使没有用过 Stream API,你是不是也能大致推断出右侧代码的含义呢?

大致说明:Stream 的核心价值

  • 声明式编程(关注做什么,而不是怎么做)
  • 链式调用,可读性强
  • 天然支持并行流
  • 减少临时变量和样板代码

二、Stream API 的整体执行流程(重点)

首先我们需要明确的是 Stream 并不是集合,而是对数据的一次性流水线处理。其大致结构是:

数据源 → 中间操作(0~N)→ 终止操作(1 个)

举个例子 ~

users.stream() // 数据源 .filter(u -> u.isValid()) // 中间操作 .map(User::getName) // 中间操作 .collect(Collectors.toList()); // 终止操作

⚠️注意:没有终止操作,Stream 不会执行


三、常用操作实战

(一)、常见中间操作

1️⃣ filter:过滤(最常用)

对字段进行过滤,是我们使用 Stream API 最为常见的需求。如果其中需要对字段进行判空,请注意:空值过滤一定要放在最前面、复杂条件可抽成方法,提高可读性。如下 ~

list.stream() .filter(Objects::nonNull) .filter(e -> e.getStatus() == 1) .collect(Collectors.toList());

2️⃣map:对象转化

典型场景:Entity → DTO、提取某个字段、类型转换。

List<Long> ids = users.stream() .map(User::getId) .collect(Collectors.toList());

3️⃣ flatMap:扁平化

和上述 map 的区别:map 是一对一,flatMap 则是一对多并拉平,详见代码:

List<String> allTags = articles.stream() .flatMap(a -> a.getTags().stream()) .distinct() .collect(Collectors.toList());

(二)、常见终止操作

1️⃣ collect:最常见

将经过处理的数据收集起来,可以转成 List, Set, 也可以转成 Map,如下:

// 1. 转 List Collectors.toList(); // 2. 转 Set Collectors.toSet(); // 3. 转 Map (注意可能出现的重复 key) Map<Long, User> map = users.stream() .collect(Collectors.toMap( User::getId, Function.identity(), (a, b) -> a // 冲突时保留第一个 ));

2️⃣ forEach vs forEachOrdered

并行流中 forEach 不保证顺序,如果需要顺序,则需要用forEachOrdered

stream.forEach(System.out::println); stream.forEachOrdered(System.out::println);

3️⃣ anyMatch / allMatch / noneMatch

这几个非常适合做校验逻辑,如:

boolean hasInvalid = users.stream() .anyMatch(u -> !u.isValid());

(三)、常见联合操作

1️⃣ 排序

list.stream() .sorted(Comparator.comparing(User::getAge).reversed()) .collect(Collectors.toList());

2️⃣ 去重(按字段)

List<User> distinct = list.stream() .collect(Collectors.collectingAndThen( Collectors.toMap(User::getId, u -> u, (a, b) -> a), m -> new ArrayList<>(m.values()) ));

3️⃣ 模糊分页(仅适合小数据量)

list.stream() .skip((page - 1) * size) .limit(size) .collect(Collectors.toList());

四、Stream API 的常见坑

❌ Stream 不能复用

每次使用都需要重新 stream() 嗷 ~

Stream<User> stream = users.stream(); stream.count(); stream.collect(...); // 抛异常

❌ 在 Stream 中修改外部变量

Stream 是函数式风格,强调无副作用 ~

int sum = 0; list.stream().forEach(e -> sum += e); // 编译失败

❌ 过度使用 Stream

有些逻辑,用 for 循环更清晰,经验法则是:简单遍历:for;数据转换、过滤、聚合:Stream

❌ 并行流 parallelStream 真的快吗

list.parallelStream().forEach(...)

不一定快,甚至可能更慢。并行流适用于数据量大、无共享状态、CPU 密集型、单次操作耗时较长的场景,而不适合IO 操作、数据量小、有锁 / 有副作用的场景嗷 ~

总结一下下班咯 ~

Stream API不是为了炫技,而是为了让代码更清晰
用得好,它是利器;用不好,就是灾难。

👉 建议:
“能用 Stream 简化的地方用 Stream,不能就老老实实写 for”

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/2/25 0:15:56

3款宝藏AI写歌工具推荐:免费好上手,小白也能玩!

人或是想打造专属旋律的普通人而言奢望既有也有音乐想写首属于自己的歌&#xff1f;以前总觉得要懂乐理、会编曲&#xff0c;门槛太高&#xff01;现在有了AI就不一样啦&#xff0c;轻松打破创作壁垒。这里精选3款优质ai歌曲生成器都是高性价比的ai音乐免费生成工具&#xff0c…

作者头像 李华
网站建设 2026/2/25 11:50:46

找靠谱OpenGL编程指南pdf的途径和使用优缺点

对于许多图形编程初学者和开发者而言&#xff0c;寻找一份权威、系统且易于获取的《OpenGL编程指南》PDF文档&#xff0c;往往是开启学习之旅的第一步。这本书被誉为“OpenGL红宝书”&#xff0c;其系统性讲解和权威性是业界公认的。获取和使用这份PDF&#xff0c;既关乎学习的…

作者头像 李华
网站建设 2026/2/16 5:36:43

MQTT调试利器:从入门到精通的完整使用手册

MQTT调试利器&#xff1a;从入门到精通的完整使用手册 【免费下载链接】MQTT-Explorer An all-round MQTT client that provides a structured topic overview 项目地址: https://gitcode.com/gh_mirrors/mq/MQTT-Explorer 你是否在物联网项目开发中遇到过这样的困扰&am…

作者头像 李华
网站建设 2026/2/25 4:32:41

C++中如何安全地将C风格字符串转化为无符号整数

在C编程中&#xff0c;将C风格字符串&#xff08;cstring&#xff09;转换为无符号整数&#xff08;uint&#xff09;是一项基础但容易出错的底层操作。它直接关系到数据处理的正确性和程序的安全性&#xff0c;错误往往源于对输入数据边界和格式的忽视。理解其原理和陷阱&…

作者头像 李华
网站建设 2026/2/17 10:01:59

可解释人工智能在软件测试中的实践与展望、

可解释AI与软件测试的融合价值 随着AI技术深度渗透测试领域&#xff0c;从自动化测试脚本生成到智能缺陷预测&#xff0c;人工智能正重塑传统测试范式。然而&#xff0c;传统黑盒AI模型&#xff08;如深度神经网络&#xff09;的决策过程不透明&#xff0c;导致测试人员面临三…

作者头像 李华