在 JDK1.8 中 , HashMap 还引入了一个新的概念,叫做负载因子(load factor),它是指哈希表中键值对的数量与数组长度的比值 。当键值对的数量超过了负载因子与数组长度的乘积时,就会触发扩容操作,HashMap 会自动将数组长度扩大一倍,并将原来的键值对重新分配到新的数组中 。这样做的目的是为了保证散列表的性能,因为当负载因子过高时,散列表的性能会急剧下降 。
文章插图
一、HashMap基础机构HashMap 由数组和链表(或红黑树)组成 。数组是 HashMap 的主体,链表和红黑树则是为了解决哈希冲突而存在的 。数组中的每个元素都是一个单向链表的头结点,每个链表都是由若干个 Node 节点组成的,每个节点都包含了键值对的信息,以及指向下一个节点的指针 。当多个键映射到同一个位置时,它们会被存储在同一个链表中(或者是同一个红黑树中) 。当链表长度超过阈值(默认为 8)时,链表就会被转换成红黑树,这样可以提高查找效率 。
在 JDK1.8 中,HashMap 还引入了一个新的概念 , 叫做负载因子(load factor),它是指哈希表中键值对的数量与数组长度的比值 。当键值对的数量超过了负载因子与数组长度的乘积时,就会触发扩容操作,HashMap 会自动将数组长度扩大一倍,并将原来的键值对重新分配到新的数组中 。这样做的目的是为了保证散列表的性能,因为当负载因子过高时 , 散列表的性能会急剧下降 。
二、HashMap的底层数据结构解答:在jdk1.8以前 , HashMa采用链表+数组,自Jdk1.8以后,HashMap采用链表+数组+红黑树 。在下图中横链(0-15)表中表示数组,竖(1-8)表示链表,在数组长度超过8之后 , hashmap将数组自动转为红黑树 。
文章插图
HashMapJDK1.8链表和红黑树转化
三、JDK1.8对hash算法和寻址算法如何优化的?1、对Hash值算法的优化
static final int hash(Object key) {int h;return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);}
有一个key的Hash_1值:【HashMap的底层数据结构】
Hash_1: 1111 1111 1111 1111 1111 1010 0111 1100
h >>> 16 // 表示对该hash值右移16位
右移后的结果Hash_2为:Hash_2: 0000 0000 0000 0000 1111 1111 1111 1111
对上述Hash_1和Hash_2的两个值进行异或Hash_1: 1111 1111 1111 1111 1111 1010 0111 1100Hash_2: 0000 0000 0000 0000 1111 1111 1111 1111=====>: 1111 1111 1111 1111 0000 0101 1000 0011 =====> 转为10进制int值,这个值就是这个key的hash值
hash算法的优化:对每个hash值,在它的低16位中,让高低16位进行异或 , 让它的低16位同时保持了高低16位的特征,尽量避免一些hash值后续出现冲突,大家可能会进入数组的同一位置 。2、对寻址算法的优化
文章插图
(p = tab[i = (n - 1) & hash]// (n-1) & hash ==> 数组里的一个位置
hash & (n-1) 效果是跟hash对n取模是一样的,但是与运算的性能要比hash对n取模要高很多 。数组的长度会一直是2的n次方,只要他保持数组长度是2的n次方 。- 寻址为什么不用取模?
也就是说,哈希表长度 length 为 2 的整次幂时,hash & (length - 1) 的计算结果跟 hash % length 一样 , 而且效率还更好 。
- 为什么不直接用 hashCode() 而是用它的高 16 位进行异或计算新 hash 值?#
寻址算法的优化:用与运算替代取模,提升性能 。(由于计算机对比取模,与运算会更快)
推荐阅读
- 如何利用Kurukshetra以交互式的方式学习如何进行安全编码
- Spring Cloud Gateway提供的简易网关实现方式,你使用过吗?
- 贮存山药的简便方法 怎么贮存山药
- 李佳琦:更大的危机!
- 2023年“口碑最好的10部电影”,《长安三万里》第6,《怪物》第2
- 买电动车48V和60V的区别有多大?有人买60V的后悔?行内人说实话
- 不用摇号,买车带京牌?记者求证:热炒的“小专车”暗藏风险
- 买红薯时,挑“细长”还是“粗圆”?
- 预制菜:底层穷人的饲料!
- 灰色丝绒连衣裙:秋季最高大上的穿搭之一,优雅又舒适,穿出气质