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

相关推荐
v维焓5 分钟前
C++(思维导图更新)
开发语言·c++·算法
jiet_h5 分钟前
深入解析KSP(Kotlin Symbol Processing):现代Android开发的新利器
android·开发语言·kotlin
_未知_开摆7 分钟前
uniapp APP端在线升级(简版)
开发语言·前端·javascript·vue.js·uni-app
花月C13 分钟前
Spring IOC:容器管理与依赖注入秘籍
java·开发语言·rpc
ylfhpy19 分钟前
Java面试黄金宝典22
java·开发语言·算法·面试·职场和发展
..过云雨44 分钟前
11. 【C++】模板进阶(函数模板特化、类模板全特化和偏特化、模板的分离编译)
开发语言·c++
风象南1 小时前
Spring Boot 实现文件秒传功能
java·spring boot·后端
橘猫云计算机设计1 小时前
基于django优秀少儿图书推荐网(源码+lw+部署文档+讲解),源码可白嫖!
java·spring boot·后端·python·小程序·django·毕业设计
黑猫Teng1 小时前
Spring Boot拦截器(Interceptor)与过滤器(Filter)深度解析:区别、实现与实战指南
java·spring boot·后端
星河浪人1 小时前
Spring Boot启动流程及源码实现深度解析
java·spring boot·后端