ThreadLocalMap 中的 key 使用强引用会出现内存泄漏吗

x33g5p2x  于2021-12-26 转载在 其他  
字(0.8k)|赞(0)|评价(0)|浏览(267)

一 弱引用和内存泄漏

有些程序员在使用 ThreadLocal 的过程中会发现有内存泄漏的情况发生,就猜测到这个内存泄漏跟 Entry 中使用了弱应用的 key 有关系,这个理解是不对的。

二 内存泄漏相关概念

Memory overflow:内存溢出,没有足够的内存提供申请者使用。

Memroy leak:内存泄漏,程序中已分配的堆内存由于某种原因未释放或无法释放,造成系统内存的浪费,导致程序运行速度减慢甚至系统崩溃等严重的后果。内存泄漏的堆积将导致内存溢出。

三 弱引用相关概念

Java 中的引用有4种类型:强、弱、软、虚。当前这个问题主要涉及强引用和弱引用。

强引用:就是我们最常用的普通对象引用,只要有强引用指向一个对象,就能表明对象还活着,垃圾回收器就不会回收这种对象。

弱引用:垃圾回收器一旦发现了只有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存。

四 如果 key 使用强引用

假设 ThreadLocalMap 中的 key 使用了强引用,那么会出现内存泄漏吗?

此时 ThreadLocal 的内存图如下:

1 假设在业务代码中使用完 ThreadLocal,threadLocal Ref 被回收了。

2 但是因为 threadLocalMap 中的 Entry 强引用了 threadLocal,造成 threadLocal无法被回收。

3 在没有手动删除这个 Entry 以及 currentThread 依然运行的前提下,始终有强引用链 threadRdf > currentThread > threadLocalMap > entry,Entry 就不会被回收(Entry 包括了 ThreadLocal 引用和 value),导致 Entry 内存泄漏。

也就是说,ThreadLocalMap 中的 key 就算使用了强引用,是无法完全避免内存泄漏的,而且情况更糟。

这种情况下,堆中任何对象都无法回收。因为都被强引用链引着。

相关文章