当前位置: 首页 > news >正文

ThreadLocal为什么会导致内存泄漏(详细讲解)

前几天看的小林coding的ThreadLocal为什么会导致内存泄漏,但是没有看的太明白,今天趁着有空,来聊聊为什么ThreadLocal会导致内存泄漏

为什么会导致内存泄漏?

  • 弱引用的 ThreadLocal key 可能被回收:
    当程序中没有强引用指向某个 ThreadLocal 实例时,这个 ThreadLocal 对象会被 GC 回收(因为它只被 ThreadLocalMap 的弱引用持有)。
//可以看到是弱引用
static class Entry extends WeakReference<ThreadLocal<?>> {/** The value associated with this ThreadLocal. */Object value;Entry(ThreadLocal<?> k, Object v) {super(k);value = v;}}
  • 但是 ThreadLocalMap 中对应的 value 却是强引用(被entry引用):
    value 是线程局部变量真正的对象实例,是强引用,GC 不会回收。

让我们进一步想:为什么value是强引用?当key被回收后,value不是也没有人引用他了吗

1. ThreadLocalMap 结构简单理解

ThreadLocalMap 是一个特殊的哈希表,里面的每个 Entry 是:

static class Entry extends WeakReference<ThreadLocal<?>> {Object value;
}
  • key 是一个弱引用,指向 ThreadLocal 对象。
  • value 是普通的强引用,指向对应线程变量的值。

2. key 被回收后,Entry 中的 key 变成 null

当没有任何强引用指向这个 ThreadLocal 实例时,GC 会回收这个 ThreadLocal 对象,Entry 里的弱引用 key 就变成了 null。

3. 但是 Entry 对象本身和 value 对象还被谁引用?

  • Entry 对象是存储在 ThreadLocalMap 的内部数组里
  • ThreadLocalMap 是线程(Thread)对象的一个成员变量,线程对象一般是强引用(线程还活着)。
  • 也就是说,整个 ThreadLocalMap 被线程强引用,Entry 也被 ThreadLocalMap 强引用
  • 由于 Entry 本身还引用着 value(value 是普通强引用),所以 value 依然是“可达”的。

4. 关键点:value 被 Entry 强引用,Entry 被 ThreadLocalMap 强引用,ThreadLocalMap 被线程强引用

所以:

线程对象(强引用)└─ ThreadLocalMap(强引用)└─ Entry(强引用)├─ key(弱引用,已回收变 null)└─ value(强引用)

value 因此不会被 GC 回收,因为根可达路径依然存在。

5. 为什么会是“孤儿”?

key 已经为 null,没有办法再通过 ThreadLocal 找到它的 value,也就是说这个 Entry 变成“垃圾”条目。

这个垃圾条目没有被清理掉,依然占用内存

6. 什么时候 value 会被回收?

  • 线程结束,ThreadLocalMap 和 Entry 都被回收。
  • 主动调用 ThreadLocal.remove() 清理 Entry,value 的强引用断开。
  • ThreadLocalMap 清理过期(key == null)的 Entry(JDK 会有清理机制,但不一定马上执行)。

总结

key 被回收后,Entry 里的 key 为 null,但 Entry 本身仍被 ThreadLocalMap 引用,而 value 是 Entry 的普通强引用,因此 value 依然可达,不会被回收。

现在对threadLocal内存泄漏的原理是不是会清晰一点。

相关文章:

  • 模拟电路的知识
  • wordpress首页调用指定ID页面内的相册
  • CSS 外边距合并(Margin Collapsing)问题研究
  • Python 流程控制语句(return、break、continue)
  • 安全有效的 C 盘清理方法
  • 水库大坝安全监测之渗流监测
  • glibc
  • ESP32-CAM识别解析QR二维码输出数据
  • SiteAzure:信箱写信提交报错
  • 已连接(connected)UDP和未连接(unconnected)UDP的区别
  • Day52 Python打卡训练营
  • JMeter + 命令行服务器端压测全流程详解
  • ARM SMMUv3命令和事件队列分析(四)
  • 确认连接的是 Redis 主节点(master),使用 SLAVEOF NO ONE 切换
  • 【ubuntu驱动安装】安装nvidia驱动和cuda环境
  • 【C语言】*与深层理解
  • JavaScript原型,原型链。
  • bisheng系列(三)- 本地部署(后端 1.2版本)
  • 消除信息屏障推动系统联动,IBMS系统成为建筑智能控制核心枢纽
  • 深入探索IIC-OLED显示技术:嵌入式仿真平台如何重塑高校教学范式——深圳航天科技创新研究院技术赋能新一代工程教育
  • 网站开发文本/八上数学优化设计答案
  • 小程序下单/长沙靠谱关键词优化服务
  • 做动态网站的用工具/seo新闻
  • wordpress4.8换成中文/黄山网站seo
  • 旅游投资公司网站建设/关键词查询爱站网
  • 注册公司需要啥资料/官网seo是什么