0

0

在Java中HashMap的put过程是怎样的_Java数据插入流程解析

P粉602998670

P粉602998670

发布时间:2026-01-17 13:34:02

|

262人浏览过

|

来源于php中文网

原创

HashMap.put()执行时先计算扰动哈希值,再通过位运算定位桶索引;若桶为空则直接插入,否则遍历链表或红黑树判断是否覆盖,链表长度≥8且数组长度≥64时树化,扩容阈值为容量×负载因子。

在java中hashmap的put过程是怎样的_java数据插入流程解析

put() 方法执行时发生了什么

Java 中 HashMap.put() 不是简单地把键值对塞进数组,而是一套带哈希计算、冲突处理和动态扩容的完整流程。核心逻辑在 JDK 8+ 中已从“链表头插”改为“尾插”,且当链表长度 ≥ 8 且桶数组长度 ≥ 64 时,会转为红黑树。

哈希值计算与索引定位的关键细节

HashMapkey.hashCode() 做了二次扰动:(h = key.hashCode()) ^ (h >>> 16),目的是让高位也参与取模运算,降低低位相同导致的哈希碰撞概率。最终桶索引通过 tab[(n - 1) & hash] 计算(n 是数组长度,必须是 2 的幂),这比取模 % n 更快,但要求容量始终是 2 的整数次幂。

  • 如果 keynull,哈希值固定为 0,总是映射到索引 0 的桶
  • 自定义类作 key 时,必须重写 hashCode()equals(),否则可能无法正确 get()
  • 哈希值相同不等于 equals() 成立,所以后续仍需调用 equals() 判断是否覆盖

链表转红黑树的触发条件容易被误读

常见误解是“只要链表长度 ≥ 8 就转树”,实际需要同时满足两个条件:table.length >= 64binCount >= 8(即遍历链表后计数达到 8)。若数组太小(如初始容量 16),即使某桶链表很长,也不会树化,而是先触发扩容。

  • 扩容阈值默认是 capacity * loadFactor(初始为 16 × 0.75 = 12)
  • 扩容后数组长度翻倍,所有元素重新哈希再分配,开销较大,应根据预估数据量合理设置初始容量
  • 红黑树节点 TreeNode 比普通 Node 占用更多内存,树化是空间换查询时间的权衡

并发场景下 put() 的危险行为

HashMap 本身不是线程安全的。多线程同时 put() 可能导致死循环(JDK 7 链表头插 + 扩容重哈希时)、数据丢失get() 返回 null 等未定义行为。这不是 bug,是设计使然。

ModelGate
ModelGate

一站式AI模型管理与调用工具

下载

立即学习Java免费学习笔记(深入)”;

  • JDK 8 中死循环问题已修复(改用尾插),但并发 put() 仍可能丢失更新
  • 如需线程安全,优先选 ConcurrentHashMap;若只是读多写少,可考虑 Collections.synchronizedMap()
  • 禁止在 foreach 或迭代器遍历时调用 put(),会抛 ConcurrentModificationException
// 示例:手动模拟 put 过程的关键判断点
int hash = hash(key.hashCode()); // 扰动哈希
int i = (table.length - 1) & hash;
Node<K,V> p = table[i];
if (p == null) {
    table[i] = newNode(hash, key, value, null); // 直接插入
} else if (p.hash == hash && Objects.equals(p.key, key)) {
    p.value = value; // 覆盖旧值
} else if (p instanceof TreeNode) {
    e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value); // 树中插入
} else {
    for (int binCount = 0; ; ++binCount) {
        if ((e = p.next) == null) {
            p.next = newNode(hash, key, value, null);
            if (binCount >= TREEIFY_THRESHOLD - 1) // -1 因为从 0 开始计数
                treeifyBin(tab, hash); // 满足条件才尝试树化
            break;
        }
        if (e.hash == hash && Objects.equals(e.key, key))
            break;
        p = e;
    }
}

真正影响性能的往往不是单次 put(),而是哈希分布不均、频繁扩容或误用线程不安全场景。这些地方一旦出问题,现象隐蔽,排查成本远高于提前设好初始容量或选对并发容器。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

254

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

1089

2024.03.01

php中foreach用法
php中foreach用法

本专题整合了php中foreach用法的相关介绍,阅读专题下面的文章了解更多详细教程。

267

2025.12.04

length函数用法
length函数用法

length函数用于返回指定字符串的字符数或字节数。可以用于计算字符串的长度,以便在查询和处理字符串数据时进行操作和判断。 需要注意的是length函数计算的是字符串的字符数,而不是字节数。对于多字节字符集,一个字符可能由多个字节组成。因此,length函数在计算字符串长度时会将多字节字符作为一个字符来计算。更多关于length函数的用法,大家可以阅读本专题下面的文章。

954

2023.09.19

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

765

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

377

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

32

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

30

2026.01.21

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

26

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.3万人学习

Java 教程
Java 教程

共578课时 | 81.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号