ThreadLocal究竟存不存在内存泄漏?

ThreadLocal的确是WeakReference弱引用,JDK垃圾回收的时候,也确实会回收掉弱引用的对象。

但是,伙伴们,请注意下图2左边的栈,ThreadLocal引用可是强引用,只要我们没主动释放ThreadLocal变量,它所引用的Entry中的ThreadLocal对象就不会被回收掉。

47068f568afc00a25f78f16d70151dcf.png

图3 内存数据模型

所以,只要我们使用的ThreadLocal变量不释放,也就是栈里的强引用一直存在,在Entry里的ThreadLocal就不会被回收,即使它是弱引用。

如果我们使用完ThreadLocal变量,手动释放ThreadLocal对象,比如把ThreadLocal对象置为null了,那么栈里对ThreadLocal对象的强引用就消失了,如果JDK垃圾回收,就会把ThreadLocal对象回收掉,Entry里的ThreadLocal的引用是弱引用,无法阻止垃圾回收。如图4所示:

ac1b89b6e9c1088e88da6e0d62807b1f.png

图4 ThreadLocal与垃圾回收

在图4中,我们能清晰看到ThreadLocal的key是可以被自动回收变成为null,但是对应的value还是被Entry引用着呢,所以value是不能被JDK垃圾回收的。

到了这里,我想伙伴们应该知道了,在线程池场景中使用ThreadLocal是有内存泄露的可能性的,原因就是线程池的核心线程Thread是循环利用的,每个线程对应的ThreadLoalMap被强引用着,所以每个线程的ThreadLoalMap不能被回收,但是ThreadLoalMap里含有多个ThreadLocal-value的Entry,虽然ThreadLocal-key是弱引用可以被垃圾回收器自动回收,但是ThreadLocal对应的value是不能被回收的,所以说有内存泄露的情况可能性。

4、如何避免线程池中使用ThreadLocal产生内存泄漏呢?

先看下ThreadLocal源码中是如何做的。

b575bb586ab79eb0743c123eb87ef0c8.png

7a957e332666c7d0aa6b7575bf0daee3.png

红框1处就是获取ThreadLocal对应的Entry,然后再从Entry获取对应value,那么在红框2处,我们能看到这个if条件,如果Entry所对应的ThreadLocal被自动回收变成null了,那这个if判断条件是不成立的,就会走到getEntryAfterMiss这个方法里,我们再来看看getEntryAfterMiss这个方法的实现。

92958bf77737b4374a02e45ecc329603.png

我们能够看到getEntryAfterMiss的逻辑,我们传进来的Entry e其实所对应的key,也就是ThreadLocal是为null的,所以一定会走到上图红线处这个条件里,会走到expungeStaleEntry这个方法里,我们再来看看expungeStaleEntry这个方法的逻辑。

83e016d5c7b70a15d2f2f74af24e2fb7.png

上图红框1处,我们能清晰的看到,会把ThreadLocal为null所对应的value设置为null,同时把对应的Entry也设置为null,同时在红框2处,会遍历所有的ThreadLocal为null的value和对应的Entry都设置为null,这样就去除了强引用,有助于被垃圾回收。

到了这里,同学们可能会说,JDK的expungeStaleEntry的方法不是会把ThreadLocal为null所对应value和Entry对象设置为null嘛,这样就可以被垃圾回收了?那在线程池的使用场景下就不会出现内存泄露的情况了啊?

其实只有在调用ThreadLocal的get、set、remove方法的时候才会触发expungeStaleEntry方法的执行,才会把ThreadLocal为null所对应的value和Entry才会设置为null。换句话说,正常的情况是不会出现内存泄露的,但是如果我们没有调用ThreadLocal对应的set、get、remove方法就不会把对应的value和Entry设置为null,这样就可能会出现内存泄露情况。

那如何避免内存泄露的情况呢?那就是我们在不使用的时候就调用一下ThreadLoca的remove方法,来加快垃圾回收,避免内存泄露。

5、ThreadLocal为什么使用弱引用?

ThreadLocal设置成弱引用,当堆栈里对指向threadlocal的强引用回收之后,就说明这个threadlocal就没用了,但是此时还有map中的key也指向了它,若是这个key是一个强引用,那么我们就无法对ThreadLocal进行回收,就会造成一个内存泄漏的问题,所以使用了弱引用来解决这个问题,只有弱引用指向的对象,在下次垃圾回收时就会被回收。大家可以结合图4好好理解下。

6、总结

最后简单总结一下,由于ThreadLocalMap包含了ThreadLocal,且线程Thread中包含变量ThreadLocalMap,因此ThreadLocalMap与Thread的生命周期是相同的,如果线程退出了,ThreadLocal自然就会被垃圾回收掉,所以不会出现内存泄漏。但在线程池中使用ThreadLocal的时候,我们还是要养成好习惯,ThreadLocal不在使用的时候调用remove方法,避免内存泄漏情况发生。

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

最后

这份清华大牛整理的进大厂必备的redis视频、面试题和技术文档

祝大家早日进入大厂,拿到满意的薪资和职级~~~加油!!

感谢大家的支持!!

image.png

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门即可获取!
存中…(img-CAnx3HZ0-1711863030600)]

《一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码》点击传送门即可获取!

  • 29
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值