【大白话说Java面试题】【Java基础篇】第23题:ConcurrentHashMap的底层原理是什么

第23题:ConcurrentHashMap的底层原理是什么

📚 回答:

  • JDK1.7 版本

    • 底层结构 :基于分段锁(Segment)+ 链表实现。
    • 核心原理
      • ConcurrentHashMap将整个数组分为多个段(Segment),每个段是一个独立的ReentrantLock控制的哈希表。
      • 每个Segment内部通过链表存储冲突的元素,锁粒度为Segment级别(默认16个段)。
      • 不同线程可以同时操作不同的Segment,从而提高了并发性能。
    • 初始化过程
      • 调用无参构造时,默认创建长度为16的Segment数组(即16个分段锁)。
      • 插入元素时,先根据key计算目标Segment的位置下标。如果该位置为空,则通过CAS(Compare-And-Swap)方式创建并插入一个Segment对象。
      • 插入具体元素时,会对目标Segment加锁,确保线程安全。
  • JDK1.8 版本

    • 底层结构 :基于数组 + 链表 + 红黑树实现,去掉了分段锁设计,改用CAS + synchronized细化锁粒度。

    • 核心原理

      • 调用无参构造时,不会立即初始化数组。首次调用put方法时才会创建长度为16的数组。
      • 插入元素时:
        • 如果目标位置为空,使用CAS方式插入新节点。
        • 如果发生哈希冲突,链表节点会以链表或红黑树的形式存储。
        • 当链表长度超过阈值(默认8)时,自动转换为红黑树,提升查询效率。
      • 锁粒度细化到头节点 :在链表或红黑树插入时,使用synchronized对链表头节点加锁,进一步提高并发性能。

      💡 代码示例

      以下伪代码展示了JDK1.8中put方法的核心逻辑:

java 复制代码
if (table == null || table.length == 0) {
    // CAS初始化数组(非构造函数)
    initTable(); 
}
int index = (n - 1) & hash;
Node<K,V> f = tabAt(tab, index);
if (f == null) {
    // CAS尝试插入新节点
    if (casTabAt(tab, index, null, newNode)) {
        break;
    }
} else {
    // 锁住头节点处理冲突
    synchronized (f) {
        if (tabAt(tab, index) == f) { // 双重检查
            if (f.hash >= 0) { // 链表节点
                // 遍历链表插入或树化
            } else { // 红黑树节点
                // 树操作逻辑
            }
        }
    }
}

💡 面试官视角

  • 面试官可能会问"为什么JDK1.8去掉分段锁?"

    答:因为分段锁虽然提升了并发性能,但锁粒度仍然较大。JDK1.8通过CAS和synchronized锁细化到头节点,进一步提升了并发性能。

  • 面试官可能会追问"红黑树的引入有什么好处?"

    答:当链表长度过长时,查询效率从O(n)提升到O(log n),显著优化了性能。


📌 专栏 :大白话说Java面试题 --- 01-Java基础篇

相关推荐
丷丩11 小时前
三级缓存下MVT地图瓦片服务性能优化策略
算法·缓存·性能优化·gis·geoai-up
_遥远的救世主_12 小时前
从一次结果集密集型查询 OOM 看 Java 服务的稳定性架构治理
java·后端
hhb_61812 小时前
Swift核心技术难点与实战案例解析
开发语言·ios·swift
m0_6294947312 小时前
LeetCode 热题 100-----25.回文链表
数据结构·算法·leetcode·链表
一楼的猫12 小时前
从工具链视角对比:番茄作家助手 vs 第三方写作辅助方案
java·服务器·开发语言·前端·学习·chatgpt·ai写作
程序leo源12 小时前
Qt窗口详解
开发语言·数据库·c++·qt·青少年编程·c#
likerhood12 小时前
Java static 关键字从浅入深
java·开发语言
猫猫的小茶馆13 小时前
【Python】函数与模块化编程
linux·开发语言·arm开发·驱动开发·python·stm32
计算机安禾13 小时前
【c++面向对象编程】第38篇:设计原则(二):里氏替换、接口隔离与依赖倒置
开发语言·c++
_院长大人_13 小时前
Java Excel导出:如何实现自定义表头与字段顺序的完全控制
java·开发语言·后端·excel