当前位置: 首页 > news >正文

Java并发实战:ConcurrentHashMap原理与常见面试题

  • 博客主页:天天困啊
  • 系列专栏:面试题
  • 关注博主,后期持续更新系列文章
  • 如果有错误感谢请大家批评指出,及时修改
  • 感谢大家点赞👍收藏⭐评论✍

前言

在上几篇文章中我已经给大家从原理上讲解了List集合和Set集合并发的一些常考知识和面试题,今天给大家要讲解的就是Map的并发

我们还是依旧摆脱无聊的文字,从代码上给大家讲解

首先我们先自己模拟一下多线程并发,看看Map是否是线程安全的

/*** @author 天天困* @date 2025/11/6*/
public class Test01 {public static void main(String[] args) {Map<String, String> map = new HashMap<>();for (int i = 1; i <= 30; i++){new Thread(() -> {map.put(Thread.currentThread().getName(), UUID.randomUUID().toString().substring(0, 5));System.out.println(map);}, String.valueOf(i)).start();}}
}

我们运行这个代码之后就会在控制台看到如下的信息:

在控制台中我们可以看到报错信息ConcurrentModificationException 异常

产生这个异常主要是由于多线程并发修改HashMap导致的。当一个线程正在遍历或打印map的内容时,另一个线程同时修改了map结构,就会触发此异常。

解决方案

方案一同步代码块

/*** @author 天天困* @date 2025/11/6*/
public class Test01 {public static void main(String[] args) {Map<String, String> map = new HashMap<>();for (int i = 1; i <= 30; i++) {new Thread(() -> {synchronized (map) {map.put(Thread.currentThread().getName(), UUID.randomUUID().toString().substring(0, 5));System.out.println(map);}}, String.valueOf(i)).start();}}
}

因为HashMap的put操作等都不是原子性,多线程同时访问会引ConcurrentModificationException 异常。那我们最先考虑到的就是通过synchronized关键字,实现互斥访问。同一时刻只有一个线程能执行synchronized代码块,避免了在遍历过程中HashMap结构被修改的情况

方案二 ConcurrentHashMap

/*** @author 天天困* @date 2025/11/6*/
public class Test01 {public static void main(String[] args) {Map<String, String> map = new ConcurrentHashMap<>();for (int i = 1; i <= 30; i++) {new Thread(() -> {map.put(Thread.currentThread().getName(), UUID.randomUUID().toString().substring(0, 5));System.out.println(map);}, String.valueOf(i)).start();}}
}

我们可以使用util包下的并发类中自带的ConcurrentHashMap来保证线程安全

final V putVal(K key, V value, boolean onlyIfAbsent) {if (key == null || value == null) throw new NullPointerException();int hash = spread(key.hashCode());int binCount = 0;for (Node<K,V>[] tab = table;;) {Node<K,V> f; int n, i, fh;if (tab == null || (n = tab.length) == 0)tab = initTable();else if ((f = tabAt(tab, i = (n - 1) & hash)) == null) {if (casTabAt(tab, i, null,new Node<K,V>(hash, key, value, null)))break;                   // no lock when adding to empty bin}else if ((fh = f.hash) == MOVED)tab = helpTransfer(tab, f);else {V oldVal = null;synchronized (f) {if (tabAt(tab, i) == f) {if (fh >= 0) {binCount = 1;for (Node<K,V> e = f;; ++binCount) {K ek;if (e.hash == hash &&((ek = e.key) == key ||(ek != null && key.equals(ek)))) {oldVal = e.val;if (!onlyIfAbsent)e.val = value;break;}Node<K,V> pred = e;if ((e = e.next) == null) {pred.next = new Node<K,V>(hash, key,value, null);break;}}}else if (f instanceof TreeBin) {Node<K,V> p;binCount = 2;if ((p = ((TreeBin<K,V>)f).putTreeVal(hash, key,value)) != null) {oldVal = p.val;if (!onlyIfAbsent)p.val = value;}}}}if (binCount != 0) {if (binCount >= TREEIFY_THRESHOLD)treeifyBin(tab, i);if (oldVal != null)return oldVal;break;}}}addCount(1L, binCount);return null;}

上述是jdk1.8以后ConcurrentHashMap中put操作的源码,从源码中我们可以看到它使用CAS+synchronized

简单插入-使用CAS

// 当桶(bucket)为空时,使用CAS无锁操作
if ((f = tabAt(tab, i = (n - 1) & hash)) == null) {// CAS原子操作,无锁化处理if (casTabAt(tab, i, null, new Node<K,V>(hash, key, value, null)))break;  // 插入成功
}

复杂操作-使用synchronized

// 当桶不为空,需要处理链表或红黑树时,使用synchronized
else {V oldVal = null;synchronized (f) {  // 只锁定当前桶的头节点// 处理链表操作、红黑树操作等复杂逻辑}
}

设计理念

  • 性能优化:大部分简单操作通过CAS完成,避免锁开销
  • 正确性保证:复杂操作使用传统锁机制确保线程安全
  • 粒度控制:只锁定必要的数据结构,提高并发度

为什么这样设计

  1. CAS的优势:无锁、低延迟、高并发
  2. CAS的局限:只能处理简单的原子操作
  3. synchronized:处理复杂的业务逻辑和数据结构变更

这样混合策略既保证了高性能、又确保了线程安全


我在上述给大家讲解ConcurrentHashMap中put操作的源码时说了一句jdk1.8以后是这样的,那难道jdk1.8之前的put逻辑和源码和jdk1.8以后不一样?没错是这样的。从这里就可以引发一道经典的面试题,接下来就是跟ConcurrentHashMap有关的面试题环节

常考面试题

Java中ConcurrentHashMap 1.7和1.8之间的区别

  • JDK1.7中HashMap采用数组+链表的数据结构,ConcurrentHashMap采用分段锁来实现高并发,分段锁的机制是每个Segment独立,最多支持16个线程并发执行也是默认的线程数量
  • JDK1.8中HashMap是数组+链表+红黑树的数据结构,优化了JDK1.7中数组扩容的方案,解决了Entry链死循环和数据丢失问题。对锁的粒度进行了优化,锁在链表或红黑树的节点级别进行,CAS用于无锁插入,synchronized仅需要处理复杂逻辑时使用,并且只锁点头节点。这样锁粒度更细,并发度更高

不加锁自己如何设计一个线程安全的HashMap?

这道面试题我希望大家可以通过我这篇的文章,自己去理解如何设计,包括我之前讲解过的一些其他文章,《Java并发List实战:CopyOnWriteArrayList原理与ArrayList常见面试题》

http://www.dtcms.com/a/577626.html

相关文章:

  • 前端FAQ: 在React中,如何优化⼤列表的渲染性能?
  • 华硕ROC奥创中心Armoury Crate服务崩溃解决办法
  • 工业软件国产替代:突破“卡脖子”,筑牢制造业升级基石
  • 大专生就业是否存在学历歧视?
  • Java 8 Stream API 进阶实战:从基础到业务落地的全解析​
  • Java117 最长公共前缀
  • 共聚焦显微镜(LSCM)的针孔尺寸标准解析
  • 长春网站优化方式投票链接制作
  • 酷炫的网站欢迎页面wordpress图片分页
  • 深入理解 flex-shrink:CSS 弹性布局中的 “收缩” 智慧
  • React+Tailwind CSS+Shadcn UI
  • 神经网络—— 优化
  • 有名的网站制怎样才能把网站宣传做的更好
  • MIPI DSI和MIPI Tx IP 的建立
  • 基于时间的 SQL 盲注-延时判断和基于布尔的 SQL 盲注
  • 个人微信公众号怎么做微网站seo完整教程视频教程
  • C++_chapter10_C++IO流类库
  • 树莓派5-docker里的ros常用命令
  • 网站地图1 500 怎么做网站推广方案及预算
  • 餐饮网站方案一个完整的网站怎么做
  • 弄一个关于作文的网站怎么做如何建立网站卖东西
  • 在Ubunutu上学习C语言(二):数组和指针
  • 成品网站源码78w78使用方法网站建设服务领域
  • ESP32内存分布全解析
  • Graph-R1:智能图谱检索增强的结构化多轮推理框架
  • java学习--可变参数
  • 相序诊断,快速响应!安科瑞户用光储防逆流无线电能表,破解您的安装难题,安全防逆流。
  • FPGA核心约束类型与语法
  • 给网站做网络安全的报价wordpress直播
  • 零基础从头教学Linux(Day 60)