在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 = 8MIN_TREEIFY_CAPACITY = 64

关键点在于:不是一插入就树化,也不是链表一长就树化——必须同时满足桶(bucket)中节点数 ≥ 8 整个 table 数组长度 ≥ 64,才会触发 treeifyBin()

  • 数组长度不足 64 时,即使某桶有 10 个冲突节点,也只会先扩容(resize),而不是树化
  • 红黑树节点是 TreeNode 类型,它继承自 Node,但额外携带了 parentleftright 等字段
  • 当树中节点数 ≤ 6 时,会退化回链表(通过 untreeify()

hash() 方法为什么二次扰动?

HashMapkey.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 等复合操作仍需外部同步保障原子性。

真正需要线程安全时,别靠“我只读不写”这种假设——只要存在任何写操作,就必须用 ConcurrentHashMapCollections.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约束提示词写法【教程】  如何在阿里云部署织梦网站?  如何做网站制作流程,*游戏网站怎么搭建?