【大白话说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基础篇

相关推荐
地平线开发者7 小时前
J6B vio scenario sample
算法
Flittly15 小时前
【AgentScope Java新手村系列】(16)从RAG到多路检索
java·spring boot·spring
小兔崽子去哪了15 小时前
Java 生成二维码解决方案
java·后端
BothSavage19 小时前
Trae远程开发中DeepSeek自定义模型4054错误的排查与修复
算法
小林ixn19 小时前
从暴力到KMP:一道题彻底搞懂字符串匹配的前世今生
算法
人活一口气20 小时前
从JVM调优到MCP协议:Java全栈技术体系深度总结与企业级架构实践
java·spring boot
烬羽21 小时前
字符串算法入门:从反转字符串到回文判断,面试不再慌
算法·面试
NE_STOP21 小时前
Vibe Coding -- 完整项目案例实操
java
荣码21 小时前
GraphRAG:普通RAG只能回答"点"的问题,我踩了4个坑才搞懂
java·python