ArrayList / HashMap / ConcurrentHashMap

这三大集合是 Java 面试和实际开发的必考点。下面我用面试视角 + 源码本质帮你快速理清它们的核心差异和底层原理。

一、ArrayList:动态数组的本质

核心:数组 + 扩容机制

1️⃣ 底层结构

transient Object\[\] elementData;

• 本质是 Object 数组

• size ≠ length(size 是实际元素个数)

2️⃣ 扩容机制(重点)

• 初始容量:

• new ArrayList<>() → 空数组(懒加载)

• 第一次 add 才扩容为 10

• 扩容公式:

int newCapacity = oldCapacity + (oldCapacity >> 1); // 1.5 倍

• 扩容代价高:要拷贝数组(System.arraycopy)

✅ 最佳实践

List list = new ArrayList<>(1000); // 预估容量,减少扩容

二、HashMap:Java 中最复杂的集合

核心:数组 + 链表 + 红黑树(JDK 8+)

1️⃣ 底层结构

Node<K,V>\[\] table

├── Node(链表)

└── TreeNode(红黑树)

2️⃣ 核心参数

参数 含义

默认容量 16

负载因子 0.75

树化阈值 链表长度 ≥ 8

树退化 ≤ 6

3️⃣ put 流程(高频面试题)

  1. 计算 hash:(h = key.hashCode()) ^ (h >>> 16)

  2. 判断 table 是否为空 → 初始化

  3. 计算下标:(n - 1) & hash

  4. 无冲突 → 直接插入

  5. 有冲突:

    • 链表 → 追加

    • 链表过长 → 树化

  6. size > threshold → 扩容(2 倍)

4️⃣ 为什么线程不安全?

• 并发 put 可能:

• 数据覆盖

• 死循环(JDK 7 及以前)

• size 不准

✅ 线程安全替代

• Collections.synchronizedMap

• ConcurrentHashMap

三、ConcurrentHashMap:高并发下的王者

核心目标:高并发 + 高吞吐

1️⃣ JDK 7 vs JDK 8(重点)

版本 实现

JDK 7 Segment 分段锁(16 把锁)

JDK 8+ CAS + synchronized(锁粒度更细)

2️⃣ JDK 8 实现原理

Node<K,V>\[\] table

├── 链表

└── 红黑树

• 锁粒度:只锁当前桶(头节点)

• 读操作:完全无锁(volatile + Unsafe)

3️⃣ put 核心流程

  1. 定位桶
  2. 桶为空 → CAS 插入
  3. 桶不为空 → synchronized 锁头节点
  4. 链表 / 红黑树插入
  5. 判断是否树化

4️⃣ size() 怎么实现的?

• 不用全局锁

• 使用 CounterCell + baseCount

• 最终值 ≈ 近似值(高并发下允许误差)

四、三者对比总结(面试必背)

对比项 ArrayList HashMap ConcurrentHashMap

数据结构 数组 数组 + 链表 + 红黑树 数组 + 链表 + 红黑树

线程安全 ❌ ❌ ✅

扩容 1.5 倍 2 倍 2 倍

锁 无 无 CAS + synchronized

适用场景 顺序存储 KV 存储 高并发 KV

五、面试高频追问(你可以提前准备)

✅ HashMap 为什么长度是 2 的幂?

→ 位运算 (n - 1) & hash 分布更均匀

✅ 为什么负载因子是 0.75?

→ 时间与空间的平衡(泊松分布)

✅ ConcurrentHashMap 能完全替代 Hashtable 吗?

→ 几乎可以,但 size() 不是强一致

✅ 为什么不用 synchronized 锁整个 Map?

→ 并发度太低,性能差

相关推荐
咖啡八杯7 小时前
GoF设计模式——策略模式
java·后端·spring·设计模式
用户1285261160215 小时前
我把祖传Java项目重构后,接口响应从3s砍到了200ms,只改了这几行代码
java
Linsk15 小时前
组件 = 模板 + 业务逻辑
java·前端·vue.js
星沉远浦16 小时前
用Gemini高效解决Java代码报错难以定位的问题
java
用户2986985301420 小时前
Word 文档字符级格式化:Java 实现方案详解
java·后端
笨鸟飞不快20 小时前
从单个服务到集群:一次完整的性能排查复盘
java·前端
荣码20 小时前
用Streamlit给AI应用套个界面,10行代码出Web页面
java·python
SamDeepThinking20 小时前
Java微服务练习方式
java·后端·微服务
朦胧之1 天前
AI 编程-老项目改造篇
java·前端·后端
程序猿大帅1 天前
别再只当调包侠了:用 Spring AI 落地 Function Calling,我被大模型硬生生砸出了三个大坑
java