从链表到红黑树:全面剖析 JDK 1.7 与 JDK 1.8 中的 HashMap 实现

在 Java 开发中,HashMap 是一个非常常用的数据结构。它的高效性和便捷性使得它在各种应用场景中被广泛使用。然而,很多开发者对 HashMap 的底层实现并不是非常了解,尤其是在 JDK 1.7 和 JDK 1.8 之间的实现差异。本文将深入探讨 HashMap 的底层实现以及相关机制。

HashMap 在 JDK 1.7 和 JDK 1.8 中的实现差异

JDK 1.7:数组 + 链表

在 JDK 1.7 中,HashMap 是通过数组和链表来实现的。每个数组元素是一个链表的头节点,当发生哈希冲突时,新元素将被添加到链表的末尾。虽然这种方式实现简单,但在链表过长的情况下,查询效率会显著下降,时间复杂度为 O(n)。

JDK 1.8:数组 + 链表 / 红黑树

为了优化查询效率,JDK 1.8 引入了红黑树。当链表长度超过 8 且数组长度大于 64 时,链表会转化为红黑树,从而将时间复杂度降至 O(logn)。这种改进显著提高了在大数据量下的查询效率。

HashMap 的新增流程

HashMap 的新增元素流程可以分为以下几个步骤:

  1. 计算哈希值:将 key 进行 hash 计算,得到哈希值。
  2. 确定位置:根据哈希值确定元素在数组中的位置。
  3. 插入元素
    • 如果位置为空,直接插入。
    • 如果位置不为空,判断是否为红黑树。如果是红黑树,则直接插入。
    • 如果是链表,判断链表长度是否超过 8 且数组长度是否大于 64。如果满足条件,则链表转化为红黑树,然后插入元素。如果不满足条件,则遍历链表进行插入。
为什么链表要升级为红黑树?

在 JDK 1.8 及以上版本中,当链表长度达到 8 且数组长度大于等于 64 时,链表会转换为红黑树。这是因为链表在长度较长时查询效率很低,时间复杂度为 O(n),而红黑树的时间复杂度为 O(logn)。引入红黑树可以大幅提高查询效率。

红黑树退化为链表的情况

当 TreeNode 的长度小于 6 时,红黑树会退化为链表。这种机制是为了在小数据量情况下降低不必要的复杂性和资源消耗。

HashMap 中使用红黑树的原因

HashMap 选择红黑树主要有以下几个原因:

  1. 二分查找树的不平衡性:二分查找树可能在极端情况下退化为链表,查询效率低。
  2. 链表查询效率低:链表长度越长,插入和查询效率越低。
  3. 红黑树效率高:红黑树的查找、插入和删除操作的效率都很高,是一种自平衡的二叉树结构。

综合考虑,红黑树是 HashMap 的最佳选择。

HashMap 重要的参数

HashMap 有两个重要的参数:容量(Capacity)和加载因子(LoadFactor)。

  1. 容量(Capacity):指 HashMap 中桶的数量,默认初始值为 16。
  2. 加载因子(LoadFactor):判定 HashMap 是否扩容的依据,默认值为 0.75。加载因子为 size / Capacity。
为什么加载因子是 0.75?

这是容量和性能之间平衡的结果。加载因子较大时,扩容频率低,但哈希冲突几率增加,性能下降。加载因子较小时,占用空间多,但哈希冲突几率小,性能较高。0.75 是一个折中的选择。

HashMap 如何解决哈希冲突?

在 JDK 1.8 中,HashMap 通过链式寻址法和红黑树解决哈希冲突。当链表长度超过 8 且数组容量大于 64 时,链表转化为红黑树,从而优化查询效率。

ConcurrentHashMap 如何保证线程安全?

在 JDK 1.7 中,ConcurrentHashMap 通过分段锁保证线程安全。而在 JDK 1.8 中,则采用了头节点加锁,结合 CAS + volatile 或 synchronized 的方式来保证线程安全。

相关推荐
数据小爬虫@2 小时前
深入解析:使用 Python 爬虫获取苏宁商品详情
开发语言·爬虫·python
健胃消食片片片片2 小时前
Python爬虫技术:高效数据收集与深度挖掘
开发语言·爬虫·python
王老师青少年编程3 小时前
gesp(C++五级)(14)洛谷:B4071:[GESP202412 五级] 武器强化
开发语言·c++·算法·gesp·csp·信奥赛
空の鱼3 小时前
java开发,IDEA转战VSCODE配置(mac)
java·vscode
一只小bit4 小时前
C++之初识模版
开发语言·c++
P7进阶路4 小时前
Tomcat异常日志中文乱码怎么解决
java·tomcat·firefox
王磊鑫4 小时前
C语言小项目——通讯录
c语言·开发语言
钢铁男儿4 小时前
C# 委托和事件(事件)
开发语言·c#
Ai 编码助手5 小时前
在 Go 语言中如何高效地处理集合
开发语言·后端·golang
小丁爱养花5 小时前
Spring MVC:HTTP 请求的参数传递2.0
java·后端·spring