0

0

Java中的并发工具类与ConcurrentHashMap

P粉602998670

P粉602998670

发布时间:2026-01-08 15:40:03

|

782人浏览过

|

来源于php中文网

原创

ConcurrentHashMap 不能用 put 替代 computeIfAbsent,因 put 无法保证初始化的原子性,而 computeIfAbsent 通过 RESERVED 状态、CAS 和分段锁确保 key 对应 value 只创建一次。

java中的并发工具类与concurrenthashmap

ConcurrentHashMap 为什么不能直接用 put 替代 computeIfAbsent

当多个线程同时尝试初始化某个 key 对应的 value(比如缓存构建、单例对象创建),直接用 put 会覆盖或丢失写入,而 computeIfAbsent 能保证只初始化一次,且整个过程原子。它底层利用了 NodeRESERVED 状态和 CAS + synchronized 分段加锁机制,不是简单“先查后 put”。

常见错误是这样写:

if (!map.containsKey(key)) {
    map.put(key, expensiveCreate());
}

这在并发下会多次执行 expensiveCreate(),且可能丢弃部分结果。正确做法是:

map.computeIfAbsent(key, k -> expensiveCreate());
  • computeIfAbsent 返回的是最终值,无论是否新创建
  • 如果 key 已存在(哪怕 value 是 null),不会触发 mappingFunction
  • mappingFunction 内部不要依赖外部可变状态,否则可能引发不可预测的竞态

ConcurrentHashMap 的 size()isEmpty() 为什么不准

这两个方法返回的是近似值,不加全局锁,只汇总各 segment / bin 的统计快照。高并发写入时,调用瞬间可能漏掉正在提交的修改,size() 可能比实际少,isEmpty() 可能返回 true 即使刚有线程 put 成功。

立即学习Java免费学习笔记(深入)”;

典型误用场景:

  • while (!map.isEmpty()) { ... } 做轮询消费 —— 可能提前退出
  • 根据 size() == 0 判断是否需要初始化 —— 条件竞争导致重复初始化

替代方案取决于用途:

EnablePPA中小学绩效考核系统2.0
EnablePPA中小学绩效考核系统2.0

无论从何种情形出发,在目前校长负责制的制度安排下,中小学校长作为学校的领导者、管理者和教育者,其管理水平对于学校发展的重要性都是不言而喻的。从这个角度看,建立科学的校长绩效评价体系以及拥有相对应的评估手段和工具,有利于教育行政机关针对校长的管理实践全过程及其结果进行测定与衡量,做出价值判断和评估,从而有利于强化学校教学管理,提升教学质量,并衍生带来校长转变管理观念,提升自身综合管理素质。

下载
  • 判断是否为空:改用 map.containsKey(someKey) 或更明确的哨兵标志
  • 获取精确大小:仅在低并发或调试时用,生产逻辑不应依赖它做控制流
  • 需要强一致性计数:搭配 LongAdder 单独维护

ConcurrentHashMap 在 JDK 8 和 JDK 9+ 的扩容行为差异

JDK 8 使用“多线程协助扩容”,迁移时其他线程发现 table 正在扩容,会主动帮着搬数据;JDK 9+(确切说是 Java 11 起)引入了 ForwardingNode 的优化,但关键变化在于:扩容阈值计算方式变了,且对 TreeBin(红黑树节点)的迁移逻辑更严格。

影响最直接的是:

  • 从 JDK 8 升级到 11+ 后,如果手动设置了初始容量和负载因子,但没考虑树化阈值(UNTREEIFY_THRESHOLD = 6),可能意外触发更多树化/退化,影响遍历性能
  • putAll() 在大集合场景下,JDK 11+ 更倾向于批量分段扩容,但若原 map 已接近阈值,可能引发连续多次扩容
  • 使用 new ConcurrentHashMap(initialCapacity) 时,JDK 8 实际分配的是 >= initialCapacity 的最小 2 的幂;JDK 11+ 仍如此,但内部 sizeCtl 初始化逻辑略有调整,极端情况下首次 put 可能多一次 resize

建议:线上服务升级 JDK 后,观察 GC 日志中的 ConcurrentHashMap 相关扩容频率,尤其关注 transferIndexbaseCount 的波动。

ConcurrentHashMap 实现简易线程安全计数器的陷阱

很多人用 map.compute(key, (k, v) -> (v == null ? 1 : v + 1)) 做计数,看似简洁,但每次调用都触发哈希查找 + CAS 尝试,高频更新下竞争激烈,性能远不如 LongAdder

真正适合用 ConcurrentHashMap 计数的场景是:key 空间稀疏、更新频次中低、需要按 key 聚合且后续要遍历或查询特定 key。

  • 高频单 key 计数(如请求总数)→ 用 LongAdderAtomicLong
  • 多 key 分桶计数(如按用户 ID 统计)→ ConcurrentHashMap 比嵌套 compute 更高效
  • 需要原子增减并返回旧值 → map.merge(key, 1L, Long::sum)compute 更语义清晰,且 merge 底层对数字类型做了少量优化

别忽略 LongAdder 的内存开销:它本质是数组+cell,每个线程争用不同 cell,但总内存占用比单个 AtomicLong 高数倍。权衡点始终是吞吐 vs 内存。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

829

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

733

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

733

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

396

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16925

2023.08.03

Golang 分布式缓存与高可用架构
Golang 分布式缓存与高可用架构

本专题系统讲解 Golang 在分布式缓存与高可用系统中的应用,涵盖缓存设计原理、Redis/Etcd集成、数据一致性与过期策略、分布式锁、缓存穿透/雪崩/击穿解决方案,以及高可用架构设计。通过实战案例,帮助开发者掌握 如何使用 Go 构建稳定、高性能的分布式缓存系统,提升大型系统的响应速度与可靠性。

27

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.4万人学习

C# 教程
C# 教程

共94课时 | 6.3万人学习

Java 教程
Java 教程

共578课时 | 43.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号