大厂面试HashMap,一定要注意这个点,很多人栽在了这儿

本文涉及的产品
可观测监控 Prometheus 版,每月50GB免费额度
注册配置 MSE Nacos/ZooKeeper,118元/月
应用实时监控服务-可观测链路OpenTelemetry版,每月50GB免费额度
简介: Hashmap是Java中最常用的集合类型,使用非常广泛。不过,有些细节问题很多人没有关注过,这也使很多人在面试时栽了跟头!比如,阿里很多团队为了考察候选人的基础,就出了这么一个面试题:为什么HashMap的初始长度和扩容长度是2的N次幂?

Hashmap是Java中最常用的集合类型,使用非常广泛。不过,有些细节问题很多人没有关注过,这也使很多人在面试时栽了跟头!比如,阿里很多团队为了考察候选人的基础,就出了这么一个面试题:为什么HashMap的初始长度和扩容长度是2的N次幂?

HashMap的数据结构



先了解一下HashMap的数据结构,在java中,数组和链表是最常用的两个基础数据结构,很多集合类都基于他们实现。HashMap也不例外,是一个链表数组,即数组和链表的结合体,当链表长度超过8时,链表转换为红黑树。

如上图,HashMap的数组元素存放了key-value键值对或者链表。当新建一个HashMap的时候,就会初始化一个数组。我们来看看java代码:

table 就是那个数组,Node是数组中的元素,Node中存放了key,value键值对。

当table中某个位置发生hash冲突时,就会自动拉链,该位置的元素也就变成了Node链表。从上图我们可以看到next成员变量,next会指向链表的下一个元素。

当我们往HashMap中put元素的时候,会先根据key的hashCode值计算出数组中对应的位置(下标),然后再把这个元素的key,value封装成Node放到对应的位置上。如果这个元素所在的位置已经存放了其他元素,那么在同一个位置上的元素将以Node链表的形式存放,新加入的Node放在链表头。从HashMap中get元素时,首先会根据key的hashCode值计算出数组中对应的位置,然后通过equals方法在对应位置的链表中找到相应的元素。所以,我们要尽量减少Hash冲突,如果每个位置上只有一个元素而不是形成链表,HashMap的效率将是最高的,时间复杂度将达到最理想的O(1)。不过现实场景中,基本都会拉链,HashMap查询过程也往往需要做链表遍历,这也导致了查询效率的下降。在Java1.8后HashMap数据结构中引入了红黑树,当链表过长时,链表会被转化为树形结构,以此来降低查询时遍历的层级和次数。


为什么长度是2的N次幂?


先了解一下哈希算法。Hash(哈希),一般翻译成散列,Hash是把任意长度的输入通过散列算法变换成固定长度的输出,该输出就是散列值(Hash值)。这种转换是一种压缩映射,也就是,散列值的空间通常远小于输入的空间,不同的输入可能会散列成相同的输出,所以不可能从散列值来确定唯一的输入值。简单的说Hash算法就是一种将无限的数据映射到有限地址空间的算法。

我们可以看到在HashMap中要找到某个元素,需要根据key的hashcode(hash值)来求得对应数组中的位置。这种将较大规模的输入数据映射到有限地址空间的算法就是hash算法(hashmap初始化时数组长度只有16,但基于链表数组的数据结构,hashmap可存储的元素数量却远大于16)。前面说过hashmap的数据结构是数组和链表的结合,所以我们希望hashmap里的元素尽量分布均匀些,尽量使每个位置上的元素数量只有一个,那么当我们用hash算法求得这个位置的时候,马上就可以取到数组上对应位置的元素,而不用再去遍历链表。

可能很多人想到了用hashcode对数组长度取模得到数组下标位置,不错,取模的方式确实可以让数据分布比较均匀。但是,有没有更好的方式呢?

Hashmap是利用位运算实现的,位运算是基于二进制进行计算的,在计算机世界里所有数据最终都要转为二进制数据,所以位运算的效率非常高。位运算效率至少比取模运算高一个数量级。从上图可以看到,Hashmap在put(get)元素时通过 (n-1) & hash 做位运算,来获取数组下标位置,其中n是数组长度。

利用key的hashcode值,和数组的长度减一(n-1)做“与”运算。使用位运算,除了效率高,还有其他玄机吗?当HashMap的容量是2的n次幂时,(n-1)的2进制也就是11111***111这样形式的,此时与key的hashcode值进行位运算时,能够充分的散列,使添加的元素能够均匀分布在数组的每个位置上,使hash碰撞几率降到最低,下面举例说明。

HashMap的长度是16(2的4次幂)时,它的二进制是10000,(n-1)的二进制是01111,与hash值的计算结果如上图所示。我们可以看到当hashmap长度是2的n次幂时,n-1与不同的hashcode值的位运算结果都不一样,添加的元素能够均匀分布在数组的不同位置,降低了hash冲突。

再看看HashMap的长度不是2的N次幂的情况,假设长度是10,它的二进制是01010,(n-1)的二进制是01001,与hash值的计算结果如上图所示。我们可以看到当HashMap容量不是2的n次幂时,n-1与不同的hashcode值的位运算结果有好几个都一样,这样在添加元素时就产生了严重的hash冲突。

所以,当数组长度为2的n次幂时,不同的key通过位运算获取的数组下标冲突的几率会比较小。冲突少了,添加元素的效率自然会更高,数据在数组上的分布也会更均匀,相应的链表长度也比较短。查询时,由于数据分布均匀,链表长度也比较平均,减少了遍历次数,查询效率也就会更高。

位运算虽然不是什么高深的知识,但是和计算机底层操作非常紧密,是进阶程序高手的必备技能。如果你有读源码的习惯,会发现JDK中有很多对位运算的应用。我们在日常开发中,不妨多用用位运算,一来可以提高程序性能,二来也可以帮自己养成对技术精深探究的习惯,让自己在技术的道路上越走越远!

本文就是愿天堂没有BUG给大家分享的内容,大家有收获的话可以分享下,想学习更多的话可以到微信公众号里找我,我等你哦。

相关文章
|
5天前
|
存储 算法 Java
面试必备!一文搞懂HashMap如何优雅处理哈希冲突
大家好,我是小米,一个积极的程序员。今天聊聊Java面试中的常见问题——“HashMap是怎么解决哈希冲突的?”。通过一个小故事,我们了解到HashMap使用链地址法(JDK 1.8前)和红黑树(JDK 1.8后)来处理哈希冲突。链地址法用链表存储冲突的元素,而红黑树在链表长度超过8时启用,提升查找效率。希望这个讲解能帮助你更好地理解HashMap的工作原理。欢迎留言讨论,关注我的公众号“软件求生”,获取更多技术干货!
23 3
|
5月前
|
存储 算法 Java
【Java集合类面试八】、 介绍一下HashMap底层的实现原理
HashMap基于hash算法,通过put和get方法存储和获取对象,自动调整容量,并在碰撞时用链表或红黑树组织元素以优化性能。
|
15天前
|
存储 Java 索引
HashMap高频面试题,让你掌握青铜回答与王者级回答,你值得拥有
《HashMap高频面试题,让你掌握青铜回答与王者级回答,你值得拥有》介绍了HashMap的实现原理、数据存储、哈希冲突处理及扩容机制。文章通过对比JDK 1.7和JDK 1.8版本,详细解析了不同版本中的链表与红黑树结构、Entry与Node的区别,以及put()方法的具体实现。特别指出JDK 1.8引入红黑树优化查询性能,并采用尾插法避免多线程环境下的链表环问题。负载因子和扩容机制确保了HashMap在不同场景下的高效运行。
31 2
|
3月前
|
存储 Java 程序员
Java面试加分点!一文读懂HashMap底层实现与扩容机制
本文详细解析了Java中经典的HashMap数据结构,包括其底层实现、扩容机制、put和查找过程、哈希函数以及JDK 1.7与1.8的差异。通过数组、链表和红黑树的组合,HashMap实现了高效的键值对存储与检索。文章还介绍了HashMap在不同版本中的优化,帮助读者更好地理解和应用这一重要工具。
80 5
|
3月前
|
存储 算法 安全
HashMap常见面试题(超全面):实现原理、扩容机制、链表何时升级为红黑树、死循环
HashMap常见面试题:红黑树、散列表,HashMap实现原理、扩容机制,HashMap的jd1.7与jdk1.8有什么区别,寻址算法、链表何时升级为红黑树、死循环
|
5月前
|
Java 索引
【Java集合类面试九】、介绍一下HashMap的扩容机制
HashMap的扩容机制包括初始容量16,以2的次方进行扩充,使用负载因子0.75判断是否扩容,以及链表长度达到阈值时转换为红黑树,以优化性能。
【Java集合类面试九】、介绍一下HashMap的扩容机制
|
5月前
|
存储 安全 Java
一天十道Java面试题----第二天(HashMap和hashTable的区别--------》sleep、wait、join)
这篇文章是关于Java面试的第二天笔记,涵盖了HashMap与HashTable的区别、ConcurrentHashMap的实现原理、IOC容器的实现方法、字节码的概念和作用、Java类加载器的类型、双亲委派模型、Java异常体系、GC如何判断对象可回收、线程的生命周期及状态,以及sleep、wait、join、yield的区别等十道面试题。
一天十道Java面试题----第二天(HashMap和hashTable的区别--------》sleep、wait、join)
|
5月前
|
存储 Java
【Java集合类面试七】、 JDK7和JDK8中的HashMap有什么区别?
JDK7中的HashMap使用数组加链表解决冲突,而JDK8增加了红黑树结构以优化链表过长时的性能,提高查找效率。
|
5月前
|
安全 Java
【Java集合类面试十五】、说一说HashMap和HashTable的区别
HashMap和Hashtable的主要区别在于Hashtable是线程安全的,不允许null键和值,而HashMap是非线程安全的,允许null键和值。
|
5月前
|
安全 Java
【Java集合类面试十三】、HashMap如何实现线程安全?
实现HashMap线程安全的方法包括使用Hashtable类、ConcurrentHashMap,或通过Collections工具类将HashMap包装成线程安全的Map。