Java面试加分点!一文读懂HashMap底层实现与扩容机制

简介: 本文详细解析了Java中经典的HashMap数据结构,包括其底层实现、扩容机制、put和查找过程、哈希函数以及JDK 1.7与1.8的差异。通过数组、链表和红黑树的组合,HashMap实现了高效的键值对存储与检索。文章还介绍了HashMap在不同版本中的优化,帮助读者更好地理解和应用这一重要工具。



哈喽大家好!今天咱们来聊聊Java中最经典的数据结构之一——HashMap!如果你是Java开发者,那你一定对它不陌生。HashMap 是我们进行键值对存储的好帮手,几乎是我们在日常开发中离不开的工具。本文会从数据结构、扩容机制、put和查找过程、哈希函数以及JDK 1.7与1.8的差异等多方面,来详细拆解一下HashMap的底层原理!Let's go~

数据结构:数组 + 链表 + 红黑树

在Java的HashMap中,底层数据结构是数组、链表、红黑树三者的组合。通过键值对的哈希映射,HashMap可以实现快速的数据存取。那么,HashMap是如何把这三种数据结构组合起来的呢?

  • 数组:这是HashMap的核心存储空间,称为table。当我们通过key来存取数据时,实际是把key通过哈希函数映射到table中的某个索引位置。
  • 链表:在HashMap中,链表主要是用来处理哈希冲突的。如果多个key被映射到了同一个数组索引,那么这些冲突的元素会被放在一个链表中,以链表形式存储。
  • 红黑树:在JDK1.8引入了红黑树,以优化链表的查找效率。若一个索引下的链表长度超过8,并且数组长度大于64,HashMap会将链表转换成红黑树。这样可以将查找的时间复杂度从O(n)降到O(log n),大幅度提升性能。

扩容情况:为什么是2的幂次方?

HashMap在扩容机制上也是独具匠心。扩容不仅影响性能,还会影响数据的分布和哈希碰撞,所以在容量和扩容机制设计上,HashMap非常讲究。

  • 默认大小和负载因子:HashMap的默认容量是16,负载因子是0.75。也就是说,当HashMap的填充度超过75%时,就会触发扩容操作,避免因为过多的哈希冲突而降低性能。
  • 扩容机制:扩容发生时,HashMap会将当前容量翻倍,并重新将所有元素重新哈希到新的数组中。
  • 容量始终是2的幂次方:HashMap的容量总是保持2的幂次方。这样设计的原因主要有以下几点:
  • 2的幂次方可以使(n-1) & hash的运算分布更均匀,减少哈希碰撞。
  • 使用位运算&替代取模操作,效率更高。

put方法的过程

HashMapput方法可以说是HashMap的精髓之一,理解它的执行过程,有助于我们掌握HashMap的存储机制。put方法主要分以下几个步骤:

  1. 判断table是否为空:如果table为空,HashMap会进行初始化操作,将容量扩充为默认大小16。
  2. 计算hash值和索引位置:通过key的hashCode值经过扰动函数处理后,再通过(n - 1) & hash计算出该元素存放的数组下标index。
  3. 检查是否有哈希冲突:检查table[index]处是否已经有节点。
  1. 如果没有节点,直接构造一个新的Node节点放入table[index]处;
  2. 如果已经有节点,说明发生了哈希冲突,进入下一步判断。
  1. 哈希冲突处理:在处理哈希冲突时,HashMap通过链表和红黑树来解决冲突。
  1. 若现有节点的key与新节点的key相同,就会用新的value覆盖原有值。
  2. 如果不相同,检查现有节点类型,如果是链表节点,则将新节点添加到链表中;如果链表长度超过阈值8且数组长度大于64,会将链表转换为红黑树。
  1. 判断是否需要扩容:当插入完成后,HashMap会检查当前容量是否超过负载因子0.75的阈值,如果超过则触发扩容。

哈希函数:扰动函数与hash计算

HashMap的哈希函数不仅仅是简单地用key.hashCode()来决定索引位置,因为直接使用hashCode()的低效与不均匀会导致大量哈希碰撞。因此,HashMap采用了一种“扰动函数”来优化哈希值的计算过程。

  • HashMap在计算key的哈希值时,先对key的hashCode()进行一次扰动,将hashCode的高16位和低16位进行异或运算。
  • 这个“扰动”能让哈希结果更加均匀分布,尽可能地减少哈希碰撞。

经过扰动处理后的哈希值,最终会通过(n - 1) & hash来计算索引位置,这样可以确保得到的索引位置始终位于数组范围内。

JDK1.7与JDK1.8的区别

在JDK1.7与JDK1.8之间,HashMap的实现有一些关键性变化:

  • 数据结构:JDK1.7中,HashMap采用了“数组+链表”的组合,而JDK1.8中则采用“数组+链表+红黑树”三者结合的结构。在JDK1.8中,当链表长度超过8且数组长度大于64时,链表会转化为红黑树以优化查找性能,避免长链表造成的性能瓶颈。
  • hash冲突处理方式:在JDK1.7中,链表插入新节点时采用的是头插法,这样做的好处是插入速度较快,但在并发情况下可能会产生死循环(例如在rehash期间)。而在JDK1.8中,链表插入时采用了尾插法,避免了并发扩容时死循环的问题。
  • 扩容过程:JDK1.8中,HashMap的扩容更为智能高效,通过高位运算决定节点位置是否发生变化。扩容时不再重新计算所有节点的哈希值,只需检查每个节点的高位,决定是否需要搬移至新数组。
  • 性能优化:JDK1.8的HashMap在多线程环境下性能优化明显,解决了JDK1.7在并发条件下扩容时可能导致的死循环问题。总体来看,JDK1.8的HashMap在结构上更为合理,更适用于高并发场景。

END

好了,这就是HashMap的底层设计和实现原理,学会这些知识之后,再遇到关于HashMap的面试题,你一定可以轻松应对!

  • 底层结构:HashMap采用数组、链表、红黑树组合的数据结构来存储键值对。
  • 扩容机制:HashMap默认负载因子为0.75,扩容时容量翻倍,始终保持2的幂次方以提高存储效率。
  • put过程:put方法主要包括判断初始化、计算hash值、解决哈希冲突、扩容等几个步骤。
  • 哈希函数:采用扰动函数,降低哈希碰撞,确保元素均匀分布。
  • JDK1.7 vs JDK1.8:1.8引入红黑树和尾插法处理冲突,避免了死循环,提高了多线程环境的安全性。

希望这篇文章能帮你更深入地理解HashMap!感谢阅读,欢迎留言讨论~

我是小米,一个喜欢分享技术的29岁程序员。如果你喜欢我的文章,欢迎关注我的微信公众号软件求生,获取更多技术干货!

相关文章
|
1天前
|
存储 安全 Java
Java 集合框架中的老炮与新秀:HashTable 和 HashMap 谁更胜一筹?
嗨,大家好,我是技术伙伴小米。今天通过讲故事的方式,详细介绍 Java 中 HashMap 和 HashTable 的区别。从版本、线程安全、null 值支持、性能及迭代器行为等方面对比,帮助你轻松应对面试中的经典问题。HashMap 更高效灵活,适合单线程或需手动处理线程安全的场景;HashTable 较古老,线程安全但性能不佳。现代项目推荐使用 ConcurrentHashMap。关注我的公众号“软件求生”,获取更多技术干货!
15 3
|
1月前
|
Java 程序员
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
77 14
|
26天前
|
Java 程序员
深入理解Java异常处理机制
Java的异常处理是编程中的一块基石,它不仅保障了代码的健壮性,还提升了程序的可读性和可维护性。本文将深入浅出地探讨Java异常处理的核心概念、分类、处理策略以及最佳实践,旨在帮助读者建立正确的异常处理观念,提升编程效率和质量。
115 1
|
27天前
|
Java 开发者 UED
深入探索Java中的异常处理机制##
本文将带你深入了解Java语言中的异常处理机制,包括异常的分类、异常的捕获与处理、自定义异常的创建以及最佳实践。通过具体实例和代码演示,帮助你更好地理解和运用Java中的异常处理,提高程序的健壮性和可维护性。 ##
49 2
|
27天前
|
Java 开发者
Java中的异常处理机制深度剖析####
本文深入探讨了Java语言中异常处理的重要性、核心机制及其在实际编程中的应用策略,旨在帮助开发者更有效地编写健壮的代码。通过实例分析,揭示了try-catch-finally结构的最佳实践,以及如何利用自定义异常提升程序的可读性和维护性。此外,还简要介绍了Java 7引入的多异常捕获特性,为读者提供了一个全面而实用的异常处理指南。 ####
50 2
|
1月前
|
运维 Java 编译器
Java 异常处理:机制、策略与最佳实践
Java异常处理是确保程序稳定运行的关键。本文介绍Java异常处理的机制,包括异常类层次结构、try-catch-finally语句的使用,并探讨常见策略及最佳实践,帮助开发者有效管理错误和异常情况。
100 5
|
30天前
|
Java 程序员 UED
深入理解Java中的异常处理机制
本文旨在揭示Java异常处理的奥秘,从基础概念到高级应用,逐步引导读者掌握如何优雅地管理程序中的错误。我们将探讨异常类型、捕获流程,以及如何在代码中有效利用try-catch语句。通过实例分析,我们将展示异常处理在提升代码质量方面的关键作用。
41 3
|
30天前
|
Java 数据库连接 开发者
Java中的异常处理机制:深入解析与最佳实践####
本文旨在为Java开发者提供一份关于异常处理机制的全面指南,从基础概念到高级技巧,涵盖try-catch结构、自定义异常、异常链分析以及最佳实践策略。不同于传统的摘要概述,本文将以一个实际项目案例为线索,逐步揭示如何高效地管理运行时错误,提升代码的健壮性和可维护性。通过对比常见误区与优化方案,读者将获得编写更加健壮Java应用程序的实用知识。 --- ####
|
30天前
|
开发框架 安全 Java
Java 反射机制:动态编程的强大利器
Java反射机制允许程序在运行时检查类、接口、字段和方法的信息,并能操作对象。它提供了一种动态编程的方式,使得代码更加灵活,能够适应未知的或变化的需求,是开发框架和库的重要工具。
43 2
|
1月前
|
Java
深入探讨Java中的中断机制:INTERRUPTED和ISINTERRUPTED方法详解
在Java多线程编程中,中断机制是协调线程行为的重要手段。了解和正确使用中断机制对于编写高效、可靠的并发程序至关重要。本文将深入探讨Java中的`Thread.interrupted()`和`Thread.isInterrupted()`方法的区别及其应用场景。
42 4