《Effective Java》解读第45条:谨慎使用Stream

第45条:谨慎使用Stream

Stream API功能强大,但过度使用会严重损害代码的可读性和可维护性。明智的做法是找到Stream与循环之间的平衡点,在合适的场景使用合适的工具。

什么时候适合使用Stream?

统一转换元素序列:批量对集合元素做相同变换(如map操作)

过滤元素序列:按条件筛选元素(如filter操作)

组合元素序列:求和、连接、求最小值等聚合操作

累积到集合:将元素收集到集合中,或按属性分组

搜索元素:查找满足条件的元素

书中"换位词"的例子:

找出字母相同、顺序不同的单词,并打印出单词数大于最低要求的单词。

恰到好处地使用Stream让代码更短更清晰:

java 复制代码
// 明智地使用Stream
try (Stream<String> words = Files.lines(dictionary)) {
    words.collect(groupingBy(word -> alphabetize(word)))
         .values().stream()
         .filter(group -> group.size() >= minGroupSize)
         .forEach(g -> System.out.println(g.size() + ": " + g));
}

alphabetize方法是获取单词字母重排后的字符串。

什么时候应该避免Stream?

滥用Stream会使程序代码更加难以读懂和维护。

以下是一些需要尽量避免使用Stream的情况:

  1. 需要读取或修改局部变量(Lambda只能读取final或effectively final的变量,无法修改任何局部变量)
  2. 需要从封闭方法中返回、break或continue(Lambda无法做到这些控制流操作)
  3. 需要抛出受检异常(Lambda无法抛出受检异常)
  4. 过度使用流的典型反例

还是上一个例子:

java 复制代码
// 过度使用Stream
try (Stream<String> words = Files.lines(dictionary)) {
    words.collect(groupingBy(word -> word.chars().sorted()
              .collect(StringBuilder::new,
                       (sb, c) -> sb.append((char) c),
                       StringBuilder::append).toString()))
         .values().stream()
         .filter(group -> group.size() >= minGroupSize)
         .map(group -> group.size() + ": " + group)
         .forEach(System.out::println);
}

注意:

chars()使用Stream会返回IntStream,如果需要char,需要进行类型转换。

java 复制代码
// 错误示例
"hello world".chars().forEach(System.out::print);
System.out.println();
// 正确示例
"hello world".chars().forEach( t -> System.out.print((char)t));

实用建议

  1. 辅助方法提升可读性
java 复制代码
// 将字母排序逻辑提取为方法
words.collect(groupingBy(word -> alphabetize(word)))
  1. 谨慎使用并行流:并行流虽然看似简单,但很少能真正提升性能,反而可能引入并发问题。
  2. 保持平衡、恰当使用:仅在有意义的情况下在新代码中使用它们,Stream本身是简化代码,而不是用来炫技。
相关推荐
小bo波38 分钟前
从"任意文件复制"深挖Java I/O:字符流与字节流的本质抉择
java·nio·io流·后端开发·文件复制
nanxun8861 天前
记一次诡异的 Docker 容器"串包"故障排查
java
唐青枫1 天前
别再乱用 StartNew:C#.NET TaskFactory 任务调度实战详解
c#·.net
用户1563068103511 天前
Day01 | Java 基础(Java SE)
java
行者全栈架构师1 天前
Maven dependency:tree 的 8 个高级用法
java·后端
行者全栈架构师1 天前
IDEA 中 Maven 项目的 15 个红色报错快速解决方法
java·后端
令人头秃的代码0_01 天前
mac(m5)平台编译openjdk
java
Artech1 天前
[MAF预定义的AIContextProvider-03]ChatHistoryMemoryProvider——赋予Agent从经验中学习的能力
ai·c#·agent·memory·maf
唐青枫2 天前
Java JDBC 实战指南:从 Connection 到事务和连接池
java
一个做软件开发的牛马2 天前
MyBatis-Plus 从零实战:完整搭建可运行 Demo,BaseMapper 零 SQL、Wrapper 条件构造、分页插件与代码生成器详解
java·后端