Java 8中Map.forEach遍历时直接修改Map会抛ConcurrentModificationException,应改用Iterator.remove()或先收集再批量操作;getOrDefault仅在key不存在时返回默认值,若key映射null则返回null。

详解Java 8中Map的forEach与getOrDefault_告别繁琐的判空逻辑

Java 8 Map.forEach 怎么写才不抛 ConcurrentModificationException

直接改原 Map 的同时用 forEach 遍历,八成会崩。这不是 forEach 的锅,是底层迭代器检测到结构被并发修改后主动抛的错。

常见错误现象:ConcurrentModificationException 在遍历时调了 map.remove(key)map.put(key, val) 后立刻触发。

getOrDefault 为什么有时返回 null 而不是默认值

不是 getOrDefault 失效了,而是你传进去的 key 本身在 map 里对应着 null 值——它只在 key 完全不存在时才兜底。

使用场景:从配置 Map 里取值,又不想每次写 map.get(k) != null ? map.get(k) : def 这种重复逻辑。

forEach + getOrDefault 组合用法的实际边界在哪

这两个方法可以一起用,但组合不等于自动安全——它们解决的是不同层面的问题:一个是遍历方式,一个是取值容错,叠加并不消除并发或 null 值隐患。

示例场景:统计订单状态分布,对每个 status 累加次数,初始值为 0:

Map<String, Integer> count = new HashMap<>();
orders.forEach(order -> {
    String status = order.getStatus();
    // ✅ 正确:先 get 再 put,避免 null 值干扰
    int curr = count.getOrDefault(status, 0);
    count.put(status, curr + 1);
});

替代方案:computeIfAbsent 和 compute 的适用时机

当你发现反复写 getOrDefault + put 时,说明该换更语义明确的 API 了。

computeIfAbsent 是懒加载式初始化的标准解法;compute 则适合需要基于旧值做计算的场景,比如计数、拼接、累加。

实际写的时候,最常被忽略的是 key 是否可能为 null、map 是否会被其他线程修改、以及 getOrDefault 的“默认”到底默认什么——它默认的只是“key 不存在”,不是“值不为空”。
本文转载于:互联网 如有侵犯,请联系zhengruancom@outlook.com删除。
免责声明:正软商城发布此文仅为传递信息,不代表正软商城认同其观点或证实其描述。