怎么理解 HashMap 的底层哈希表实现
怎么理解 HashMap 的底层哈希表实现

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈
谈到HashMap,很多人知道它快,但快从何来?其底层实现,本质上是用“数组 + 链表/红黑树”这套组合拳,模拟了一张逻辑上的散列表。核心目标非常明确:把任意一个键(Key)快速、准确地映射到一个固定的内存位置,从而实现平均接近 O(1) 的存取效率。这听起来简单,背后的设计却充满了权衡与巧思。
数组是主干,“桶”决定落点
一切的基础是一个名为 Node 的数组。你可以把它想象成一排排的“桶(bucket)”。数据并非按顺序存放,而是由键的哈希值来决定它的“家”在哪。举个例子,键 "user123" 经过自身的 hashCode() 方法计算,再经过HashMap内部的扰动函数处理,得到一个最终的哈希值,假设是 1987。接着,用这个值对当前数组长度取模(比如 1987 % 16 = 3),结果 3 就是它的归宿——索引为3的那个桶。这个过程,就是所谓的“哈希函数定位”,是高效访问的起点。
冲突不可避免,链表和红黑树来兜底
理想很丰满,现实却难免碰撞。不同的键完全有可能计算出相同的数组下标,这就是“哈希冲突”。比如 "abc" 和 "def" 都落到了索引5的桶里。怎么办?覆盖显然不行。于是,HashMap采用了链式结构:后来的节点就接在已有节点之后,形成一个链表。
- 在插入方式上,JDK 7采用的是头插法,而JDK 8改为了尾插法。这一改动主要是为了解决并发扩容时可能引发的环形链表问题,提升了稳定性。
- 那么,链表会不会太长导致查找变慢?当然会。因此,当链表长度增长到 ≥ 8 并且 此时数组的总长度也 ≥ 64 时,链表会自动升级为红黑树。这个数据结构能将查找性能从链表的 O(n) 提升到 O(log n)。
- 反过来,如果因为删除操作导致树中的节点数减少到 ≤ 6,红黑树又会退化成链表。这避免了在数据量较小时,维护树结构带来的额外开销。
看,这里全是平衡的艺术:在空间、时间和实现复杂度之间寻找最佳实践。
键的比较必须成对重写:hashCode() + equals()
HashMap如何判断两个键是“相同”还是“不同”?这个过程是两步走的严谨校验:
- 首先比较
hashCode():如果两个键的哈希值不同,那么它们肯定不是同一个键,可以直接放入新的位置或链表末尾。 - 如果哈希值相等(发生了哈希碰撞),事情就没那么简单了,这时需要请出
equals()方法进行内容的深度比较:返回 true,则视为重复键,新值会覆盖旧值;返回 false,则视为不同的键,会在链表或树中新增一个节点。
这就引出一个至关重要的实践规则:当你打算用自定义的类对象作为HashMap的键时,必须同时、一致地重写 hashCode() 和 equals() 方法。否则,即使两个对象的内容完全一致,也会因为默认的Object类方法(比较内存地址)而被判定为不同的键,导致数据重复存储,这显然不是我们想要的结果。
扩容不是小事,影响性能的关键环节
HashMap不是一成不变的,它需要成长。初始容量默认是16,加载因子默认是0.75。这意味着,当桶数组中的元素数量达到 容量 × 加载因子(即 16 * 0.75 = 12)时,扩容机制就会触发。
- 扩容时,会创建一个容量为原来两倍的新数组(例如从16扩大到32)。
- 接下来是一个相对耗时的操作:遍历旧数组中所有的节点,为每个节点基于新数组长度重新计算哈希和下标,然后将它们迁移到新数组对应的新“桶”中。
- 这个过程对于链表和红黑树节点同样适用,红黑树在迁移过程中还会进行必要的平衡调整。
正因为扩容涉及大量的数据重新分配,对性能有直接影响,所以如果能在初始化时就预知大致的数据量规模,建议直接指定一个合理的初始容量(例如 new HashMap(1024)),这样可以有效减少后续扩容的次数,提升整体性能。
游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。
同类文章
SpringBoot2.7.x将logback升级到1.3.x以上版本的全过程解析
SpringBoot2 7 x将logback升级到1 3 x以上版本的全过程解析 不少开发者在尝试将SpringBoot 2 7 x项目中的Logback升级到1 3 x或更高版本时,都会遇到一个典型的启动报错。这背后的原因其实很明确:SpringBoot 2 7 x默认依赖的是logback-c
Xrender支持哪些图形格式
xrender支持的图形格式 核心说明 首先得澄清一个常见的误解:xrender本身并不是一个图像解码库。它实际上是X Window System的一个渲染扩展,主要负责提供抗锯齿、路径绘制、渐变、合成这些高级的2D渲染能力。那么,图片是怎么显示出来的呢?通常,应用程序会先用其他专门的库(比如处理P
ubuntu中copendir命令如何与其他命令组合使用
在Ubuntu中组合使用文件复制命令 在Ubuntu系统中,你可能听说过copiodir这个命令,但事实上它并不存在。你真正需要掌握的是功能强大且无处不在的cp命令,它是Linux系统中文件和目录复制的核心工具。那么,如何让cp命令与其他命令协同工作,实现更高效的自动化文件管理呢?关键在于灵活运用管
怎样用nginx日志解决跨域问题
如何通过Nginx配置解决跨域问题:从原理到实战 开门见山地说,试图直接利用Nginx日志来解决跨域问题,这个思路本身存在误区。Nginx日志的核心作用是什么?它本质上是一个“记录系统”,负责详尽记录每一次访问详情与错误信息,但其本身并不具备主动配置或修复跨域问题的能力。跨域问题的根源在于浏览器的同
Debian系统phpstorm的内存设置
Debian 下 PhpStorm 内存设置指南 想让 PhpStorm 在 Debian 上跑得更快更稳?内存配置是关键一步。下面这份指南,将帮你从修改核心参数到验证生效,一步步搞定。 一 修改 vmoptions 文件 动手之前,记得先关闭正在运行的 PhpStorm。接下来,打开终端,找到并编
- 日榜
- 周榜
- 月榜
1
2
3
4
5
6
7
8
9
10
相关攻略
2015-03-10 11:25
2015-03-10 11:05
2021-08-04 13:30
2015-03-10 11:22
2015-03-10 12:39
2022-05-16 18:57
2025-05-23 13:43
2025-05-23 14:01
热门教程
- 游戏攻略
- 安卓教程
- 苹果教程
- 电脑教程
热门话题

