333. Java Stream API - 按年份找出合作最多的作者对:避免 Optional.orElseThrow() 的风险

333. Java Stream API - 按年份找出合作最多的作者对:避免 Optional.orElseThrow() 的风险

在之前的代码中,我们使用 orElseThrow()Optional 中取出值:

java 复制代码
map -> map.entrySet().stream()
          .max(Map.Entry.comparingByValue())
          .orElseThrow()

✅ 这在数据完整的情况下没问题,但一旦某一年没有任何文章或没有两人合作文章,就会抛出 NoSuchElementException,这在 按年份分组 时变得危险。


✅ 更安全的写法:保留 Optional

我们改用 Optional 来安全地包裹最大值:

java 复制代码
Collector<PairOfAuthors, ?, Optional<Map.Entry<PairOfAuthors, Long>>> pairOfAuthorsEntryCollector =
    Collectors.collectingAndThen(
        Collectors.groupingBy(
            Function.identity(),
            Collectors.counting()
        ),
        map -> map.entrySet().stream()
                  .max(Map.Entry.comparingByValue()) // ⚠️ 注意:不再 orElseThrow()
    );

这个 Collector 的返回类型现在是 Optional<Map.Entry<...>>,避免了直接解包的风险。


📦 构建 flatMapping Collector

我们继续包一层 flatMapping,用于提取作者对:

java 复制代码
Collector<Article, ?, Optional<Map.Entry<PairOfAuthors, Long>>> flatMapping =
    Collectors.flatMapping(
        toPairOfAuthors,
        pairOfAuthorsEntryCollector
    );

这样按年份分组后的结果是:

java 复制代码
Map<Integer, Optional<Map.Entry<PairOfAuthors, Long>>>

⚠️ 问题:Optional 的 Map 无法直接使用

这类 Map<Integer, Optional<...>> 类型虽然安全,但不实用 ------ 空值也占据空间。

我们希望清理掉空值,得到:

java 复制代码
Map<Integer, Map.Entry<PairOfAuthors, Long>>

🎯 解决方案:用 flatMap 清洗 Optional

我们使用 Optional.map().stream() + flatMap() 组合:

java 复制代码
Map<Integer, Map.Entry<PairOfAuthors, Long>> histogram =
    articles.stream()
            .collect(
                Collectors.groupingBy(
                    Article::inceptionYear,
                    flatMapping
                )
            ) // 得到 Map<Integer, Optional<...>>
            .entrySet().stream()
            .flatMap(entry -> 
                entry.getValue()
                     .map(value -> Map.entry(entry.getKey(), value))
                     .stream() // Optional 转 Stream(为空时返回空流)
            )
            .collect(Collectors.toMap(
                Map.Entry::getKey,
                Map.Entry::getValue
            ));

🔍 分析这段关键 flatMap 逻辑:

java 复制代码
.flatMap(entry ->
    entry.getValue()                      // Optional<Map.Entry<...>>
         .map(value -> Map.entry(entry.getKey(), value)) // Optional<Map.Entry<Integer, Map.Entry<...>>>
         .stream()                        // 转成 Stream
)
  • 如果 Optional 是空的,map() 返回空 Optional,.stream() 就是空流 ✅
  • 如果有值,生成一个新的 (year, authorPair) 条目
  • flatMap() 中,空的条目自动被过滤

🧪 小示例:Optional 扁平化

java 复制代码
Map<Integer, Optional<String>> map = Map.of(
    1, Optional.empty(),
    2, Optional.of("two"),
    3, Optional.empty(),
    4, Optional.of("four")
);

Map<Integer, String> map2 = map.entrySet().stream()
    .flatMap(entry -> entry.getValue()
                           .map(value -> Map.entry(entry.getKey(), value))
                           .stream()
    )
    .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue));

map2.forEach((k, v) -> System.out.println(k + " :: " + v));

⏱️ 输出结果:

java 复制代码
2 :: two
4 :: four

🚫 空的 Optional 被自动过滤,无需显式判断。


🧠 技巧总结

技术点 含义与应用
Optional.map() 对 Optional 内部值做变换,如果是 empty 则什么也不做
Optional.stream() Java 9+ 中的新方法,将 Optional 转为 Stream
Stream.flatMap() 扁平化多个 stream(在这里是 Optional)为一个连续的流
collect(Collectors.toMap()) 将流重新收集为 Map

🎓 类比讲解

Optional.stream() 的类比 : 把 Optional 想象成一扇门:

  • 如果门后有人(有值),就放他进来(stream 里有元素);
  • 如果没人(空),就关门不说话(空 stream);

flatMap() 就像一个过滤器,只让真正进门的人留下。

相关推荐
小码哥_常3 小时前
安卓开发秘籍:解锁10大性能优化秘诀
前端
小码哥_常3 小时前
MyBatis-Plus:让数据库操作飞起来的神器
后端
2301_811274314 小时前
基于SpringBoot的智能家居管理系统
spring boot·后端·智能家居
AI人工智能+电脑小能手4 小时前
【大白话说Java面试题】【Java基础篇】第15题:JDK1.7中HashMap扩容为什么会发生死循环?如何解决
java·开发语言·数据结构·后端·面试·哈希算法
舒一笑4 小时前
我把设备指纹生成逻辑拆开了:它到底凭什么区分不同设备?
后端·程序员·掘金技术征文
try2find4 小时前
打印ascii码报错问题
java·linux·前端
Nicander4 小时前
多数据源下@transcation事务踩坑
java·后端
郑州光合科技余经理4 小时前
同城O2O海外版二次开发实战:从支付网关到配送算法
开发语言·前端·后端·算法·架构·uni-app·php
冰暮流星5 小时前
javascript事件案例-全选框案例
服务器·前端·javascript
sjsjsbbsbsn5 小时前
大模型核心知识总结
java·人工智能·后端