ThreadLocal的原理是操作Thread内部的一个ThreadLocalMap,这个Map的Entry继承了WeakReference,设值完成后map中是(WeakReference,value)这样的数据结构。Java中的弱引用在内存不足的时候会被回收掉,回收之后变成(null,value)的形式,key被收回掉了。
如果这个线程执行完之后销毁,value也会被回收,这样也不会出现内存泄露。但如果是在线程池中,线程 执行完后不被回收,而是返回线程池中。此时Thread有个强引用 指向 ThreadLocalMap,ThreadLocalMap有强引用 指向 Entry,导致Entry中key为null的value无法被回收,一直存在内存中。在执行了ThreadLocal.set()方法之后一定要记得使用ThreadLocal.remove(),将不要的数据移除掉,避免内存泄漏。
通过分析源码,java也做了一定优化,即使出现了上述的(null,value)情况,再调用一次ThreadLocal.set()也可以将这个废弃的替代调用
这是java8种ThreadLocal.set()方法,for循环是遍历整个Entry数组,红色框的地方是碰到了(null,value)的处理逻辑。也就是碰到了内存泄漏后,会将原来的Entry替换掉,避免内存泄漏。