ConcurrentHashMap.computeIfAbsent 是线程安全懒加载缓存的首选方法,它原子性地完成“查-算-存”,避免竞态条件和重复初始化;要求 mappingFunction 非 null,禁用阻塞逻辑,慎防死锁,支持 Supplier 延迟求值。

ConcurrentHashMap.computeIfAbsent 是 Java 中实现线程安全懒加载缓存最简洁、高效的方式之一。它在键不存在时才执行计算逻辑,并自动将结果放入 map,整个过程原子完成,无需手动加锁或双重检查。
手动判断再插入存在竞态条件:多个线程同时发现 key 不存在,都去计算并 put,造成重复构造和覆盖风险。而 computeIfAbsent 内部基于 CAS 和锁分段(JDK8+ 使用 synchronized + Node 锁)保证“查-算-存”三步原子性,天然规避重复初始化问题。
适合缓存开销大、构造耗时、且实例可复用的对象(如 JSON 解析器、正则 Pattern、数据库连接配置等):
ConcurrentHashMap<String, ObjectMapper> mapperCache = new ConcurrentHashMap<>();
ObjectMapper getMapper(String configKey) {
return mapperCache.computeIfAbsent(configKey, key -> {
// 只有首次调用才会执行,后续直接返回缓存值
ObjectMapper om = new ObjectMapper();
om.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
return om;
});
}
computeIfAbsent 的 mappingFunction 不允许返回 null(会抛 NPE),所以构造逻辑里要确保返回有效对象。另外,该方法在计算期间会短暂阻塞同桶内其他写操作,因此:
立即学习“Java免费学习笔记(深入)”;
computeIfPresent + 单独标记机制当构造逻辑需要外部参数或上下文时,可封装为 Supplier,延迟到真正需要时才触发:
ConcurrentHashMap<String, Supplier<DataSource>> dsCache = new ConcurrentHashMap<>();
DataSource getDataSource(String url) {
return dsCache.computeIfAbsent(url, u ->
() -> createDataSourceWithRetry(u) // 真正的创建逻辑被延迟执行
).get(); // 调用 get() 才真正构造
}
这种方式把“是否创建”的决策交给 computeIfAbsent,“何时创建”的控制权留给调用方,更灵活也更安全。
基本上就这些。用好 computeIfAbsent,既能写出简洁代码,又能避开并发陷阱,是 Java 懒加载缓存的推荐起点。
以上就是Java里如何使用ConcurrentHashMap.computeIfAbsent构建懒加载缓存_Java懒加载策略讲解的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号