news 2026/6/10 11:35:37

jdk1.7 在多线程下扩容可能导致的死循环问题

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
jdk1.7 在多线程下扩容可能导致的死循环问题

首先先看看 hashmap 在jdk1.7 下扩容的核心方法

void transfer(Entry[] newTable) { Entry[] src = table; int newCapacity = newTable.length; // 遍历旧数组的每一个格子(桶) for (int j = 0; j < src.length; j++) { Entry<K,V> e = src[j]; if (e != null) { src[j] = null; // 断开旧数组的引用,虽然不是必须的,但有助于GC // 遍历链表,把链表上的节点一个一个摘下来,重新放到新数组里 do { // 【关键点1】先保存下一个节点,防止待会链表断了找不到后面的人 Entry<K,V> next = e.next; // 计算这个节点在新数组中的下标 int i = indexFor(e.hash, newCapacity); // 【关键点2】头插法核心:将当前节点 e 的 next 指向新数组位置原本的头节点 e.next = newTable[i]; // 【关键点3】将新数组该位置的头指针指向 e(e 变成了新的头节点) newTable[i] = e; // 【关键点4】继续处理旧链表的下一个节点 e = next; } while (e != null); } } }

可以看到,jdk1.7 版本下的扩容依靠的是头插法实现元素搬移

头插法 的特点:假设旧数组某个位置的链表是A -> B -> C。 因为代码逻辑是:每次把新来的元素插到最前面(e.next = newTable[i]然后newTable[i] = e)。 所以搬运到新数组后,顺序会倒置,变成C -> B -> A

为什么JDK 1.7用头插法? 设计者当时认为,后插入的数据被访问的概率更高(热点数据),放在头部查得快。但这在多线程下埋下了祸根。

那其实在多线程的情况下就会有问题了,举个例子:

假设有两个线程Thread 1Thread 2同时在扩容。 旧链表结构:A -> B。 线程 1 执行到你注释的那一行挂起了(Paused):

Entry<K,V> next = e.next; // 线程1在这里卡住了! // 此时对于线程1来说: e = A, next = B
接下来可能发生的事情:
  1. 线程 2 抢占 CPU 完成了扩容
    • Thread 2 把AB都搬到了新数组。
    • 注意:因为是头插法,顺序反了
    • 在新内存里,现在的链表结构是:B -> A(B 指向 A,A 指向 null)。
  1. 线程 1 醒来继续执行
    • Thread 1 里的变量还停留在挂起前的状态:e = Anext = B
    • Thread 1 开始干活:
      • 将 A 指向 B
      • 头节点变成 A
      • 此时 A 指向 B,B 指向 A(死循环开始)
  1. 线程 1 接着处理 B,B 头插当头节点,此时 B 指向 A,A 指向 B
  2. 然后接着处理 A,A 头插当头节点,此时 A 指向 B,B 指向 A
  3. 然后循环往复 A,B 轮流当头节点,死循环

如图:


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

终极指南:利用智能计数技术实现高效目标分析

面对复杂场景下目标数量统计的挑战&#xff0c;智能计数技术正成为解决密度检测和对象分析难题的关键工具。本指南将带您深入了解如何通过先进算法实现精准的目标识别与数量统计&#xff0c;让数据分析工作变得前所未有的高效和准确。 【免费下载链接】X-AnyLabeling Effortles…

作者头像 李华
网站建设 2026/5/30 20:47:52

5个技巧让Open WebUI数据可视化效果翻倍:新手也能轻松上手

5个技巧让Open WebUI数据可视化效果翻倍&#xff1a;新手也能轻松上手 【免费下载链接】open-webui Open WebUI 是一个可扩展、功能丰富且用户友好的自托管 WebUI&#xff0c;设计用于完全离线操作&#xff0c;支持各种大型语言模型&#xff08;LLM&#xff09;运行器&#xff…

作者头像 李华
网站建设 2026/6/8 19:40:32

计算机毕业设计springboot基于JAVA的作业管理系统 基于SpringBoot+Java的在线作业发布与批阅平台 JavaWeb轻量级作业调度与成绩反馈系统

计算机毕业设计springboot基于JAVA的作业管理系统r14735cq &#xff08;配套有源码 程序 mysql数据库 论文&#xff09; 本套源码可以在文本联xi,先看具体系统功能演示视频领取&#xff0c;可分享源码参考。高校日常教学中&#xff0c;作业依旧是检验学生掌握程度、巩固知识点的…

作者头像 李华
网站建设 2026/6/6 7:04:40

计算机毕业设计springboot民宿信息展示与管理及可视化平台系统 基于SpringBoot的民宿数据可视化与智能运营平台 融合大数据的民宿资源聚合及多维分析系统

计算机毕业设计springboot民宿信息展示与管理及可视化平台系统b3c1b6t4 &#xff08;配套有源码 程序 mysql数据库 论文&#xff09; 本套源码可以在文本联xi,先看具体系统功能演示视频领取&#xff0c;可分享源码参考。 民宿行业在疫情后迎来爆发式增长&#xff0c;但信息分散…

作者头像 李华
网站建设 2026/6/8 9:15:11

如何快速掌握OpenUtau:虚拟歌手音乐制作的完整指南

如何快速掌握OpenUtau&#xff1a;虚拟歌手音乐制作的完整指南 【免费下载链接】OpenUtau Open singing synthesis platform / Open source UTAU successor 项目地址: https://gitcode.com/gh_mirrors/op/OpenUtau 想要创作属于自己的虚拟歌手歌曲却不知从何开始&#x…

作者头像 李华
网站建设 2026/6/9 21:32:37

终极指南:5分钟快速掌握macOS与Android USB网络共享

在移动办公成为常态的今天&#xff0c;如何快速将Android手机的移动网络共享给Mac电脑是许多用户的迫切需求。HoRNDIS驱动程序正是为此而生的专业解决方案&#xff0c;它能通过USB数据线将Android设备变身为网络适配器&#xff0c;为macOS用户提供稳定可靠的网络连接。 【免费下…

作者头像 李华