news 2026/5/1 4:46:34

ConcurrentHashMap 扩容后如何保证数据一致性?深度解析并发迁移与内存可见性机制

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
ConcurrentHashMap 扩容后如何保证数据一致性?深度解析并发迁移与内存可见性机制

在高并发场景下,ConcurrentHashMap(CHM)的扩容不仅要快,更要保证数据一致性—— 不能丢数据、不能读到中间状态、不能出现幻读或重复读。JDK 1.8 通过精心设计的并发迁移流程 + 内存屏障 + 原子操作,实现了强一致性写 + 弱一致性读的平衡。本文将深入剖析其一致性保障机制。


一、核心挑战:扩容期间如何避免数据不一致?

假设线程 A 正在 put,线程 B 正在 get,同时线程 C 触发了扩容。此时可能出现以下问题:

  1. 读线程读到“半迁移”状态(部分桶已迁,部分未迁);
  2. 写线程插入到旧表,但读线程去新表查,导致“丢失”
  3. 多个线程同时迁移同一桶,造成数据覆盖或重复

ConcurrentHashMap通过以下四大机制解决这些问题。


二、一致性保障机制详解

✅ 1. ForwardingNode:引导读写操作到正确位置

这是一致性最关键的基石

  • 当某个桶(bucket)的数据被迁移到新 table 后,原桶位置立即被替换为ForwardingNode(hash = -1);
  • ForwardingNode持有对新 table 的引用
  • 任何线程访问该桶时:
    • 若发现是ForwardingNode立即跳转到新 table 对应位置继续操作
    • 写操作(put/remove)会协助完成剩余迁移
    • 读操作(get)直接在新 table 中查找。
// get 操作中处理 ForwardingNode Node<K,V> f = tabAt(tab, i); if (f != null) { if (f.hash == MOVED) // MOVED = -1 return getNode(f.nextTable, key); // 跳转到新表 // ... 正常查找 }

🔒效果

  • 无论桶是否迁移完成,所有操作都能定位到最新数据所在位置
  • 避免“旧表查不到、新表还没写”的数据丢失问题。

✅ 2. 桶级别迁移 + synchronized 锁头节点

  • 迁移一个桶时,先对原桶的头节点加synchronized
  • 确保同一时间只有一个线程能迁移该桶
  • 迁移完成后,才将原桶设为ForwardingNode
synchronized (f) { // f 是原桶头节点 if (tabAt(tab, i) == f) { // 双重检查 // 迁移链表/红黑树到 nextTab setTabAt(nextTab, i, newHead); // 标记原桶已迁移 setTabAt(tab, i, fwd); } }

🛡️作用

  • 防止多个线程重复迁移同一桶;
  • 保证迁移过程的原子性;
  • 避免 put 操作在迁移中途插入旧表导致数据丢失。

✅ 3. volatile + Unsafe 保证内存可见性

CHM 大量使用volatileUnsafevolatile 语义写(putObjectVolatile),确保多线程间的内存可见性

  • table字段是volatile
  • ForwardingNode的设置通过setTabAt()(内部调用Unsafe.putObjectVolatile);
  • nextTableForwardingNode中也是final(构造即可见)。
static final <K,V> void setTabAt(Node<K,V>[] tab, int i, Node<K,V> v) { U.putObjectVolatile(tab, ((long)i << ASHIFT) + ABASE, v); }

💡意义
一旦一个线程将桶设为ForwardingNode,其他线程立即可见,不会因缓存读到旧值而误操作旧表。


✅ 4. 扩容期间的 put 操作:自动重定向 + 协助迁移

当 put 操作发现桶是ForwardingNode,它会:

  1. 调用helpTransfer()协助完成当前扩容
  2. 然后重新执行 put 流程(此时 table 已更新为新表);
  3. 最终数据一定写入新 table 的正确位置
else if (f.hash == MOVED) tab = helpTransfer(tab, f); // 协助迁移并返回新 table // 循环重新尝试 put

结果
即使扩容正在进行,所有新写入的数据都会进入新表,不会残留在旧表中。


三、读操作的一致性:弱一致但不错误

CHM 的get 操作不加锁,因此提供的是弱一致性(weakly consistent)

  • 可能读到扩容前的旧值(如果迁移尚未完成);
  • 绝不会读到“损坏”或“中间状态”的数据
  • 也不会抛出ConcurrentModificationException

这是因为:

  • 所有 Node 的keyhashval(除 compute 系列外)都是final 或 volatile
  • 链表/红黑树结构在迁移时是整体替换,不会出现“断链”;
  • ForwardingNode确保读操作总能找到数据(无论新旧表)。

📌注意
弱一致性 ≠ 不一致!它只是不保证“实时最新”,但保证读到的一定是某个合法历史状态


四、扩容完成后的切换:原子更新 table 引用

当所有桶迁移完毕,最后一个完成任务的线程会:

  1. nextTable赋值给table
  2. 清空nextTable
  3. 重置sizeCtl为新的阈值。

由于tablevolatile,这一切换对所有线程立即可见,后续操作自然使用新表。


五、总结:CHM 如何做到“又快又稳”?

机制作用一致性保障
ForwardingNode引导操作到新表防止读写错位
synchronized 锁桶头串行化迁移防止并发迁移冲突
volatile / Unsafe内存可见性确保状态变更及时同步
put 自动重试写入新表避免数据残留旧表
弱一致读高性能无锁保证读到合法状态

💬一句话总结
“通过 ForwardingNode 实现无缝跳转,通过细粒度锁保证迁移原子性,通过内存屏障确保可见性——三者结合,让扩容既高效又安全。”

视频看了几百小时还迷糊?关注我,几分钟让你秒懂!(发点评论可以给博主加热度哦)

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 16:23:34

开题报告 springboot和vue 学生作品发布平台 优秀学生作品分享系统

目录 系统背景与目标技术架构核心功能模块创新点应用场景 项目技术支持可定制开发之功能亮点源码获取详细视频演示 &#xff1a;文章底部获取博主联系方式&#xff01;同行可合作 系统背景与目标 SpringBoot与Vue技术栈结合的学生作品发布平台旨在为高校学生提供一个展示、分享…

作者头像 李华
网站建设 2026/5/1 4:45:43

内存数据库SQL入门:如何实现高速读写与实时分析

内存数据库 SQL 将数据存储在系统主内存中&#xff0c;通过标准 SQL 接口进行操作&#xff0c;从而极大提升数据访问速度。与传统基于磁盘的数据库相比&#xff0c;它消除了I/O瓶颈&#xff0c;使得实时数据处理和分析成为可能。在当前高并发、低延迟的应用需求下&#xff0c;内…

作者头像 李华
网站建设 2026/4/18 9:11:26

开题报告 springboot和vue_校内闲置书籍交易网站 二手书交易系统

目录 技术选型背景系统核心功能模块技术实现细节创新点与优化方向预期成果 项目技术支持可定制开发之功能亮点源码获取详细视频演示 &#xff1a;文章底部获取博主联系方式&#xff01;同行可合作 技术选型背景 SpringBoot和Vue是目前主流的前后端分离开发框架组合。SpringBoo…

作者头像 李华
网站建设 2026/4/26 3:42:54

掌控发动机“心脏”精度:蓝光3D扫描在凸轮轴全尺寸检测中的应用

汽车发动机凸轮轴具有连续曲面&#xff08;如凸轮桃升程曲线&#xff09;、复杂集成特征&#xff08;轴颈/相位角/油孔等&#xff09;及严苛公差要求。采用蓝光三维扫描技术&#xff0c;通过非接触全尺寸扫描、秒级动态采集与智能全局拼接&#xff0c;可满足凸轮轴高精度、高效…

作者头像 李华
网站建设 2026/4/12 11:02:16

[信息论与编码理论专题-23]:《信息简史》简介

《信息简史》&#xff08;英文原名&#xff1a;The Information: A History, a Theory, a Flood&#xff09;是美国著名作家、科技史学者詹姆斯格雷克&#xff08;James Gleick&#xff09;于2011年出版的一部广受赞誉的非虚构作品。这本书以“信息”为核心&#xff0c;跨越数千…

作者头像 李华