ThreadLocal内存泄漏原因以及避免方案

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

ThreadLocal内存泄漏原因以及避免方案_第1张图片
这是java8种ThreadLocal.set()方法,for循环是遍历整个Entry数组,红色框的地方是碰到了(null,value)的处理逻辑,也就是碰到了内存泄漏后会将原来的Entry替换掉避免内存泄漏。

你可能感兴趣的:(转载)