0

0

Java里的弱哈希映射WeakHashMap原理_自动清理过期项的机制

P粉602998670

P粉602998670

发布时间:2026-03-13 09:01:02

|

840人浏览过

|

来源于php中文网

原创

WeakHashMap 的 key 会被自动回收,因为其 key 用 WeakReference 包装,GC 发现无强引用时会在下次 GC 后清除对应 entry;value 为强引用,若 value 反向持有 key 会导致内存泄漏。

java里的弱哈希映射weakhashmap原理_自动清理过期项的机制

WeakHashMap 的 key 为什么会被自动回收?

因为 WeakHashMap 的 key 是用 WeakReference 包装的,GC 一旦发现该 key 没有强引用指向它,就会在下一次 GC 时清掉这个 entry——不是“定时清理”,也不是“懒加载扫描”,就是靠 GC 触发的被动回收。

常见错误现象:WeakHashMap.get(key) 突然返回 null,但你没调 remove();或者遍历 entrySet() 时发现 size 越来越小,甚至变 0。

  • 使用场景:缓存对象生命周期完全依赖外部引用(比如 GUI 组件映射配置、监听器上下文绑定),不希望缓存本身阻止对象被回收
  • key 必须是可被 GC 的对象,String 字面量、Integer 小值常量池对象通常不会被回收(有强引用驻留),慎用
  • value 仍是强引用,如果 value 又反过来持有 key(比如内部类、lambda 捕获),会形成引用链,导致 key 无法被回收

WeakHashMap 和 HashMap 在 put/get 行为上有什么差异?

表面用法几乎一样,但底层行为差异直接决定是否“漏数据”:put 后 key 可能在任意 GC 后消失;get 之前必须确保 key 还活着,否则查不到。

参数差异:没有额外构造参数控制“弱强度”,所有 key 统一走 WeakReference;而 ReferenceQueue 是内部私有使用的,不暴露给用户。

立即学习Java免费学习笔记(深入)”;

  • 不要依赖 size() 做逻辑判断——它只反映当前未被 GC 的 entry 数,不是“已插入数”
  • containsKey(key) 可能返回 false,即使刚 put 过,只要 key 已被 GC
  • 遍历时推荐用 entrySet().iterator(),避免并发修改异常;但注意迭代中 key 可能中途被回收,next() 返回的 EntrygetKey() 可能为 null

为什么 WeakHashMap 不清理 value 引用?会导致内存泄漏吗?

会。value 是强引用,如果 value 是一个大对象、或持有对 key 的反向引用(如匿名内部类实例),就会让 key 无法被 GC,使 WeakHashMap 失去“弱性”。这不是 bug,是设计使然——它只保证 key 弱,不干涉 value。

Joker AIx
Joker AIx

一站式AI创意生产平台,覆盖图像、视频、音频、文案全品类创作

下载

典型泄漏场景:用 WeakHashMap<View, Bitmap> 缓存图片,但 Bitmap 内部持有了 View 的引用(比如通过 setTag() 或自定义回调)。

  • 检查 value 是否间接引用了 key,尤其是通过闭包、监听器、Handler、静态内部类等方式
  • 必要时手动置空 value 中对 key 的引用,或改用 SoftReference 包装 value(需自行封装)
  • 别指望 WeakHashMap 替你管理 value 生命周期;它只解决 key 的悬挂问题

WeakHashMap 的实际清理时机和性能影响

entry 清理发生在 GC 后的“后处理阶段”:GC 把 key 对应的 WeakReference 加入内部关联的 ReferenceQueue,然后 WeakHashMap 在后续的 getputsize 等方法中惰性地遍历 queue 并清理 table 中对应槽位——所以不是 GC 完就立刻干净,可能有延迟。

性能影响:每次操作都可能触发 queue 清扫,最坏情况要遍历整个 queue(虽然通常很短);高并发写入时,WeakHashMap 没有分段锁,全表锁竞争比 ConcurrentHashMap 更明显。

  • 高频读写场景下,WeakHashMap 的吞吐量明显低于 HashMap
  • 不要在 tight loop 里反复 get() 同一个 key,万一它刚被 GC,每次都会触发 queue 扫描
  • 如果需要确定性清理,自己维护 ReferenceQueue + 定期清扫逻辑,而不是依赖 WeakHashMap 的惰性机制

真正难的是平衡“弱引用”的语义和实际业务里的引用关系。很多人以为用了 WeakHashMap 就万事大吉,结果 value 一牵扯,key 就卡死在内存里——这时候看堆 dump,往往发现 WeakHashMap$Entry 还在,但它的 key 字段是 null,而 value 却牢牢占着几百 MB。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1030

2023.08.02

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

254

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

1089

2024.03.01

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1567

2023.10.24

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

215

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

192

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

61

2026.01.05

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

443

2023.07.18

Python异步编程与Asyncio高并发应用实践
Python异步编程与Asyncio高并发应用实践

本专题围绕 Python 异步编程模型展开,深入讲解 Asyncio 框架的核心原理与应用实践。内容包括事件循环机制、协程任务调度、异步 IO 处理以及并发任务管理策略。通过构建高并发网络请求与异步数据处理案例,帮助开发者掌握 Python 在高并发场景中的高效开发方法,并提升系统资源利用率与整体运行性能。

36

2026.03.12

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.2万人学习

Java 教程
Java 教程

共578课时 | 81万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号