0

0

在Java里HashMap如何存储键值对_Java哈希表实现机制解析

P粉602998670

P粉602998670

发布时间:2026-02-08 13:34:02

|

358人浏览过

|

来源于php中文网

原创

HashMap底层是数组+链表+红黑树的混合结构:哈希值经扰动运算后与(长度-1)按位与得索引;冲突时用链表,满足≥8节点且数组≥64则转红黑树;null键固定存于tab[0];重写equals必须重写hashCode以保证哈希一致;非线程安全,并发下易出错。

在java里hashmap如何存储键值对_java哈希表实现机制解析

HashMap底层用数组加链表/红黑树存储键值对

Java的HashMap不是单纯哈希表,而是一个“数组 + 链表 + 红黑树”的混合结构。它把key.hashCode()经过扰动运算后取模,映射到数组索引位置;冲突时先用链表串连节点,当链表长度≥8且数组长度≥64,就转为红黑树优化查找性能。

常见误解是“哈希值直接当索引”,其实会做两步处理:(h = key.hashCode()) ^ (h >>> 16)(高位参与运算),再& (table.length - 1)(等价于取模,但要求table.length必须是2的幂)。

  • 数组长度永远是2的幂(如16、32、64),扩容时翻倍
  • 链表转红黑树有双重条件:链表节点数≥8 table.length >= 64,缺一不可
  • 红黑树退化回链表的阈值是节点数≤6(不是7)

put()过程中如何计算hash、定位桶、处理冲突

调用map.put(key, value)时,核心流程是:算hash → 定位tab[i] → 若为空直接插入;否则遍历链表/树,检查key.equals();相同则覆盖value;不同则尾插或树插入。

注意keynull是特例:它固定被哈希为0,存在数组第0个位置(tab[0]),且只允许一个null键。

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

  • hash()函数对null返回0,对非null对象执行高位异或扰动
  • 数组索引计算用hash & (length - 1),不是%,所以初始容量必须是2的幂
  • 链表插入是头插(JDK 7)还是尾插(JDK 8+)?JDK 8起统一为尾插,避免多线程扩容时死循环

为什么重写equals()必须重写hashCode()

如果两个对象equals()返回true,但hashCode()不同,它们会被分配到HashMap不同桶里,导致get()永远找不到——因为查找时先比哈希值,哈希不等直接跳过后续equals()判断。

DeepSider
DeepSider

浏览器AI侧边栏对话插件,集成多个AI大模型

下载

典型错误:只重写equals(),没同步更新hashCode()逻辑,尤其在使用IDE自动生成时漏掉字段。

  • 所有参与equals()比较的字段,都必须参与hashCode()计算
  • 推荐用Objects.hash(f1, f2, f3)生成hashCode(),安全且简洁
  • 字段含可变对象(如List)时,要确认其hashCode()行为稳定(比如用ImmutableList

并发场景下HashMap可能崩溃的三个关键点

HashMap本身不保证线程安全,多线程put()可能引发数据丢失、死循环(JDK 7)、或树化异常(JDK 8)。根本原因在于resize()和节点插入都未加锁,且依赖共享状态(如sizemodCount、链表指针)。

即使只读操作,在扩容中途也可能看到不一致的结构(如部分桶已迁移,部分未迁移)。

  • JDK 7中多线程put可能触发环形链表,get()陷入死循环
  • JDK 8中虽改用尾插,但并发resize仍可能导致节点丢失或重复
  • ConcurrentHashMap才是正确选择:JDK 8起用synchronized锁单个桶,而非全局锁

实际用的时候,别只盯着“存得快”,更要盯住hashCode()一致性、null键边界、以及是否真需要并发安全——很多所谓“并发问题”,其实是误用了非线程安全容器。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

阿里巴巴推出的全能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语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

241

2023.09.22

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

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

641

2024.03.01

length函数用法
length函数用法

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

931

2023.09.19

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

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

613

2023.08.10

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

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

283

2025.12.24

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

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

21

2026.01.21

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

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

23

2026.01.21

C# 多线程与异步编程
C# 多线程与异步编程

本专题深入讲解 C# 中多线程与异步编程的核心概念与实战技巧,包括线程池管理、Task 类的使用、async/await 异步编程模式、并发控制与线程同步、死锁与竞态条件的解决方案。通过实际项目,帮助开发者掌握 如何在 C# 中构建高并发、低延迟的异步系统,提升应用性能和响应速度。

18

2026.02.06

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

61

2026.02.06

热门下载

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

精品课程

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

共23课时 | 3.3万人学习

C# 教程
C# 教程

共94课时 | 8.9万人学习

Java 教程
Java 教程

共578课时 | 60.2万人学习

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

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