Threadlocal的作用是实现线程隔离,定义一个threadlocal的静态变量,被多个线程引用,其中每个线程都拥有一个独特的副本进行操作,互不可见
- 啥是弱引用?
就是一个对象是弱引用时,System.gc()会直接回收掉这个地方,强引用的话,则GC不会回收掉,例如下面的Demo中,第二次weak.get()返回值是null,因为他是弱引用
public static void main(String[] args) {
String abc=new String("abc");
WeakReference<String> weak=new WeakReference<String>(abc);
System.out.println("print:" +weak.get());
abc=null;
System.gc();
System.out.println("print:" +weak.get());
}
弱引用有什么好处?
强引用的对象GC的要求严格,必须没有引用的情况下才可以回收(GC Roots不可达),Java设计了一种短暂生命周期的对象,弱引用对象,只在创建和FGC之前有效,FGC之后,不管有没有继续引用,都直接释放内存的一种机制。
帮助更好的管理内存。ThreadLocal中 弱引用的使用
- 每个Thread类都有一个ThreadLocalMap
- 每个ThreadLocalMap里面都有一个 Entry[] table
- 每个Entry就是一个键值对, key 就是ThreadLocal而且是弱引用的,Value是普通对象,强引用
重点注意:Entry不是弱引用,只有key是弱引用,而value是强引用
ThreadLocal 和OOM
问题来了,如果一个线程不停的给不同thradLocal属性设置值,并且该线程长期不结束,那么这些threadLocal对象会在fgc的时候被GC掉,但是value值并不会被GC掉,因为是线程.ThreadLocalMap.Entry.value 在强引用,所以这些value值,既不能被访问到(key已经被回收了),又无法被回收,而且所有引用了这个ThreadLocal的线程里,都有这个现象。所以容易造成OOMThreadLocal和线程池
线程池里面的ThreadLocal会出错吗?
会的,所以新任务启动的时候,要init,把所有threadlocal属性都置为null,调用 remove()函数