在Java中HashMap底层是如何实现的_Java哈希映射结构解析
发布时间 - 2026-01-26 00:00:00 点击率:次HashMap底层是数组+链表+红黑树,Java 8起当链表长度≥8且数组长度≥64时转为红黑树,否则扩容;hash()二次扰动缓解低位哈希冲突;put过程含哈希计算、桶定位、冲突处理与可能的树化或扩容;非线程安全,多线程put会导致数据覆盖或死循环。
HashMap 的底层数据结构是数组 + 链表 + 红黑树
Java 8 开始,HashMap 不再只是“数组 + 链表”,而是在链表长度 ≥ 8 且数组长度 ≥ 64 时,将链表转为红黑树。这个阈值由两个条件共同控制:TREEIFY_THRESHOLD = 8 和 MIN_TREEIFY_CAPACITY = 64。
关键点在于:不是一插入就树化,也不是链表一长就树化——必须同时满足桶(bucket)中节点数 ≥ 8 且 整个 table 数组长度 ≥ 64,才会触发 treeifyBin()。
- 数组长度不足 64 时,即使某桶有 10 个冲突节点,也只会先扩容(resize),而不是树化
- 红黑树节点是
TreeNode类型,它继承自Node,但额外携带了parent、left、right等字段 - 当树中节点数 ≤ 6 时,会退化回链表(通过
untreeify())
hash() 方法为什么二次扰动?
HashMap 对 key.hashCode() 做了位运算扰动:
static final int hash(Object key) {
int h;
return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
}这是为了把高位也参与低位的索引计算,缓解哈希值低位相似导致的聚集问题(比如 HashMap 存的是 Integer,且数值集中在小范围,原始 hashCode 就是自身值,低位重复率高)。
如果不扰动,仅用 (n - 1) & hash 计算下标,那么 n 是 2 的幂次(如 16),n-1 就是 0b1111,只取 hash 低 4 位——高位完全被丢弃,容易造成大量碰撞。
put() 过程中如何处理哈希冲突与扩容
调用 put(K, V) 时,核心流程是:计算 hash → 定位桶(tab[i = (n-1) & hash])→ 若桶为空则直接新建 Node;否则遍历链表或树节点比对 key.equals()。
- 如果找到相同 key(
hash相等且equals()
为 true),则覆盖 value,并返回旧值
- 如果没找到,新节点插入链表尾部(JDK 8 后不再是头插,避免多线程扩容死链)
- 插入后若链表长度达到 8,且数组长度 ≥ 64,则树化;否则若
size >= threshold(默认 0.75 × capacity),触发 resize() - 扩容时新数组长度翻倍,原节点根据 hash 的新增 bit 位决定留在原索引还是落到
i + oldCap
为什么 HashMap 不是线程安全的?典型表现是什么
多个线程同时 put 可能引发两种典型问题:
- 数据覆盖:两个线程计算出同一桶位置,都判断该桶为空,各自新建 Node 写入,后者直接覆盖前者
-
死循环(JDK 7):多线程扩容时头插法导致链表成环,
get()遍历时无限循环(JDK 8 改为尾插,消除了该问题,但并发写仍不安全)
注意:ConcurrentHashMap 并非简单加锁整个 map,而是采用分段锁(JDK 7)或 CAS + synchronized 锁单个桶(JDK 8+),粒度更细。但即便如此,computeIfAbsent 等复合操作仍需外部同步保障原子性。
真正需要线程安全时,别靠“我只读不写”这种假设——只要存在任何写操作,就必须用 ConcurrentHashMap、Collections.synchronizedMap(),或明确加锁。
# java
# node
# 为什么
# Integer
# 循环
# 数据结构
# 继承
# 线程
# 多线程
# map
# 并发
# table
# 链表
# 红黑
# 为空
# 加锁
# 的是
# 这是
# 是在
# 多个
# 才会
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
javascript日期怎么处理_如何格式化输出
Edge浏览器怎么启用睡眠标签页_节省电脑内存占用优化技巧
Laravel如何安装Breeze扩展包_Laravel用户注册登录功能快速实现【流程】
如何获取免费开源的自助建站系统源码?
Laravel如何与Pusher实现实时通信?(WebSocket示例)
免费的流程图制作网站有哪些,2025年教师初级职称申报网上流程?
Laravel怎么实现模型属性转换Casting_Laravel自动将JSON字段转为数组【技巧】
html如何与html链接_实现多个HTML页面互相链接【互相】
html5如何实现懒加载图片_ intersectionobserver api用法【教程】
东莞市网站制作公司有哪些,东莞找工作用什么网站好?
如何在 Pandas 中基于一列条件计算另一列的分组均值
Laravel如何部署到服务器_线上部署Laravel项目的完整流程与步骤
百度输入法ai面板怎么关 百度输入法ai面板隐藏技巧
,怎么在广州志愿者网站注册?
如何用免费手机建站系统零基础打造专业网站?
zabbix利用python脚本发送报警邮件的方法
Laravel怎么清理缓存_Laravel optimize clear命令详解
Laravel怎么实现微信登录_Laravel Socialite第三方登录集成
Laravel怎么实现API接口鉴权_Laravel Sanctum令牌生成与请求验证【教程】
如何实现javascript表单验证_正则表达式有哪些实用技巧
Laravel如何清理系统缓存命令_Laravel清除路由配置及视图缓存的方法【总结】
网站图片在线制作软件,怎么在图片上做链接?
详解Android——蓝牙技术 带你实现终端间数据传输
如何续费美橙建站之星域名及服务?
如何在阿里云购买域名并搭建网站?
网站优化排名时,需要考虑哪些问题呢?
Python文件操作最佳实践_稳定性说明【指导】
如何用搬瓦工VPS快速搭建个人网站?
网站广告牌制作方法,街上的广告牌,横幅,用PS还是其他软件做的?
如何选择PHP开源工具快速搭建网站?
Laravel如何使用Eloquent进行子查询
进行网站优化必须要坚持的四大原则
利用 Google AI 进行 YouTube 视频 SEO 描述优化
Laravel的Blade指令怎么自定义_创建你自己的Laravel Blade Directives
网站建设要注意的标准 促进网站用户好感度!
七夕网站制作视频,七夕大促活动怎么报名?
如何快速搭建高效香港服务器网站?
如何在IIS服务器上快速部署高效网站?
iOS验证手机号的正则表达式
制作企业网站建设方案,怎样建设一个公司网站?
谷歌浏览器下载文件时中断怎么办 Google Chrome下载管理修复
Laravel Eloquent关联是什么_Laravel模型一对一与一对多关系精讲
微信小程序 canvas开发实例及注意事项
iOS UIView常见属性方法小结
如何用花生壳三步快速搭建专属网站?
UC浏览器如何设置启动页 UC浏览器启动页设置方法
手机软键盘弹出时影响布局的解决方法
Claude怎样写约束型提示词_Claude约束提示词写法【教程】
如何在阿里云部署织梦网站?
如何做网站制作流程,*游戏网站怎么搭建?


