从链表到红黑树:全面剖析 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 的方式来保证线程安全。

相关推荐
C-SDN花园GGbond1 小时前
【探索数据结构与算法】插入排序:原理、实现与分析(图文详解)
c语言·开发语言·数据结构·排序算法
迷迭所归处2 小时前
C++ —— 关于vector
开发语言·c++·算法
架构文摘JGWZ3 小时前
Java 23 的12 个新特性!!
java·开发语言·学习
leon6253 小时前
优化算法(一)—遗传算法(Genetic Algorithm)附MATLAB程序
开发语言·算法·matlab
Navigator_Z3 小时前
数据结构C //线性表(链表)ADT结构及相关函数
c语言·数据结构·算法·链表
拾光师4 小时前
spring获取当前request
java·后端·spring
aPurpleBerry4 小时前
neo4j安装启动教程+对应的jdk配置
java·neo4j
锦亦之22334 小时前
QT+OSG+OSG-earth如何在窗口显示一个地球
开发语言·qt
我是苏苏4 小时前
Web开发:ABP框架2——入门级别的增删改查Demo
java·开发语言
姜太公钓鲸2334 小时前
c++ static(详解)
开发语言·c++