Thread有一个ThreadLocal.ThreadLocalMap类型的threadLocals成员变量,他是一个map,底层是一个ThreadLocalMap.Entry的数组,每个ThreadLocal变量都有一个threadLocalHashCode,根据这个hashcode可计算出当前ThreadLocal在数组的index值。
ThreadLocalMap.Entry是一个弱引用类型WeakReference<ThreadLocal<?>>,Entry的value就是我们需要存储的对象,而key则是一个弱引用的referent,当ThreadLocal没有任何强引用时,GC后即使还有弱应用存在也会被回收,这时Entry的key为null。所以即使线程一直活着,只要ThreadLocal不存在任何强引用,就会被回收器回收,避免内存泄漏。

clipboard.png

那么问题来了,即使Entry的key为弱引用会被回收,但value一直以key为null的Entry保存在map中,这是强引用,永远不会被回收。ThreadLocalMap的解决方法是在每次get、set、remove时都会将key为null的entry清除掉,这样就会被垃圾回收器清理了。

为什么不将entry的value也设置为弱引用呢,这是因为如果value是弱引用,执行GC后,若value被清除,那我们永远也拿不到ThreadLocal保存的数据了。

通常实践中,我们常把ThreadLocal做为单例使用,因为我们要保证key为ThreadLocal的值全局都能访问到。当不是单例使用情况下,我们要记得在不实用ThreadLocal时调用一下ThreadLocal.remove()方法,删除key。help gc.


Degaulle
26 声望6 粉丝

more thinking, less BB