史上最全面的HashMap详解,HashMap实现原理与基础知识点介绍
创作时间:
作者:
@小白创作中心
史上最全面的HashMap详解,HashMap实现原理与基础知识点介绍
引用
CSDN
1.
https://blog.csdn.net/i_YOLO/article/details/145643361
1.HashMap实现原理
(1)JDK1.7以前
在JDK1.7版本之前,HashMap底层数据结构采用的是
数组+链表
的形式
当向HashMap中添加数据时,会采用哈希算法,将元素的key映射到底层数组对应的位置,我们称之为槽位(Bucket)
众所周知,
哈希算法对于不同key的计算结果可能相同
,所以当不同的key映射到数组上的同一个槽位时,它们就会以链表的形式存储到同一个槽位上
由于链表的查询速度时O(n),所以冲突很严重,当一个索引上的链表非常长时,查询效率就会很低
(2)JDK1.7以后
所以在JDK1.8版本的时候做了优化,HashMap采用
数组+链表和数组+红黑树
两种方法相结合的方式存储数据
当链表长度达到规定的阈值(默认为8)时,会将链表转化为红黑树存储,提高查询效率,时间复杂度降低到O(logn)
同时,在数量小于6时,会将红黑树转化为链表
2.HashMap的
put(key,value)
方法(JDK1.8版本)
- HashMap采用put(key,val)方法存储对象,调用hashCode()方法计算哈希值,获取对应bucket的位置
- 如果该bucket上没有存放元素,则直接插入
- 如果该bucket上存放着元素,则
比较key值 - 如果key值相同,则直接覆盖value
- 如果key值不同,则需要遍历链表或者红黑树来查询是否有相同的key
- 查询到相同的key,则覆盖value
- 查询不到相同的key,对于链表采取尾插法插入数据,对于红黑树则插入到树的末尾
- 存放数据完毕后,检查链表长度是否超过阈值(默认为8)
- 如果链表长度超过阈值,同时
HashMap底层数组的长度超过64
,则将链表转化为红黑树,以提高查询效率
- 检查负载因子是否超过阈值(默认为0.75)
- 如果超过阈值,则进行
扩容
操作(负载因子概念和扩容机制详细看第3大点)
- 需要扩容的话完成扩容操作,数据添加到此结束
3.HashMap的扩容机制
(1)负载因子
负载因子是衡量哈希表使用情况的重要指标,定义如下:
负载因子 = 数组中存储的元素个数 / 数组大小
负载因子太低会导致大量的空桶浪费空间,负载因子太高会导致大量的碰撞,降低性能
0.75的负载因子在这两个因素之间取得了良好的平衡。
(2)扩容机制
HashMap默认的负载因子为0.75,即如果HashMap中的元素个数超过了总容量的75%,则会触发扩容机制:
- 将哈希表(底层数组)扩大到原来的两倍,创建一个新的数组
- 将旧数组中的数据移到新数组中
4.HashMap在多线程下存在的问题
JDK1.7中的HashMap使用头插法插入元素,在多线程的环境下,扩容的时候有可能导致
环形链表
的出
现,形成死循环。因此,JDK1.8使用尾插法插入元素,在扩容时会保持链表元素原本的顺序,不会出现
环形链表的问题。
多线程同时执行put操作
,如果计算出来的索引位置是相同的,那会造成前一个key被后一个key覆
盖,从而导致元素的丢失。此问题在DK1.7和DK1.8中都存在。
热门推荐
书法精髓:掌握横竖撇捺的力度与韵律
霹雳舞进了巴黎奥运会?你你你你要跳舞吗~
丢失金首饰后应如何处理?
宇文恺:隋朝杰出的城市规划与建筑大师
点墨成金 宁波探索石墨烯产业化科创新路径
铁血丹心杨邦乂
铁血丹心杨邦乂
6亿人近视,青少年占一半!抓住孩子近视前的“预警信号”
八字命理:丑未相冲详解及化解方法
宝器是什么意思
华工在美国的血泪史——石泉大屠杀
揭秘陈良全,从基层干部到改革巨擘的人生航程
如何利用数据分析提升供应链管理优化效果
论文参考文献为什么使用 GB/T 7714-2015 格式?有什么特别之处嘛?
防暑有妙招!濮阳中医教你自制凉茶!
现代通信技术专业学什么
ICU:重症加强护理病房的全面解析
2025年度“国考”如何报名?需要具备什么条件?有何特点?一起了解→
如何合理了解和掌握基金收费标准?这些基金收费标准的制定依据有哪些方面?
AI大模型,职教在探索
携号转网攻略:从流程到套餐选择全解析
糖的热量及营养成分
小苏打清洁妙用大全
常见的人民调解方法有哪几种
初中作文多少分算高分?评分标准及写作技巧全解析
手术后眼睛疼痛是什么原因
去除饭盒异味小妙招:告别臭味,让饭盒焕然一新
海洋科普 | 世界上最大的海龟——棱皮龟
中国龟类大全:30多种珍稀物种的详细介绍
中国印章-孙溟展浅析《孔子庙堂碑》