多线程002--ThreadLocal有哪些内存泄露问题,如何避免

每个Thread都有一个ThreadLocal.ThreadLocalMap的map,该map的key为ThreadLocal实例,它为一个弱引用。

弱引用有利于GC回收。

弱引用与软引用的区别在于:只具有弱引用的对象拥有更短暂的生命周期。

在垃圾回收器线程扫描它所管辖的内存区域的过程中,一旦发现了只具有弱引用的对象,不管当前内存空间足够与否,都会回收它的内存。

不过,由于垃圾回收器是一个优先级很低的线程,因此不一定会很快发现那些只具有弱引用的对象。

当ThreadLocal的key等于null时,GC就会回收这部分空间,但是value却不一定能够被回收,因为他还与Current Thread存在一个强引用关系。

多线程002--ThreadLocal有哪些内存泄露问题,如何避免

 

由于这个强引用关系,会导致value无法回收。

如果这个线程对象不会销毁,那么这个强引用关系会一直存在,就会出现内存泄露情况。

所以只要这个线程对象能够及时被GC回收,就不会出现内存泄漏。

 

解决办法:

使用ThreadLocalMap种的setEntry()、getEntry(),遇到key为null的情况,会对value设置为null。

也可以显式调用ThreadLocal的remove()方法进行处理。

 

总结:

ThreadLocal不是用于解决共享变量的问题的,也不是为了协调线程同步的,而是为了方便每个线程处理自己的状态而引入的一个机制。

每个Thread内部都有一个ThreadLocal.ThreadLocalMap类型的成员变量,该成员变量用来存储实际的ThreadLocal变量副本。

ThreadLocal并不是为线程保存对象的副本,它仅仅起到一个索引的作用。它的主要目的是为每一个线程隔离一个类的实例,这个实例的作用范围仅限于线程内部。

上一篇:《Java多线程编程核心技术第二版》读书笔记


下一篇:【ThreadLocal】