threadlocal本质的作用就是线程之间对变量操作的隔离
我们看看它内部是怎么实现的
set方法:
public void set(T value) {
Thread t = Thread.currentThread();
ThreadLocalMap map = getMap(t);
if (map != null)
map.set(this, value);
else
createMap(t, value);
}
首先是创建了一个ThreadLocalMap:
ThreadLocalMap getMap(Thread t) {
return t.threadLocals;
}
map.set(this, value);this表示的是当前threadlocal引用,value是我们要存的值,在set的过程中将当前threadlocal引用作为key,value作为map传入一个entiy对象:
tab[i] = new Entry(key, value);
这个entry是个弱引用:
static class Entry extends WeakReference<ThreadLocal<?>> {
/** The value associated with this ThreadLocal. */
Object value;
Entry(ThreadLocal<?> k, Object v) {
super(k);
value = v;
}
}
注意在entry中没有保存当前threadlocal引用,而是将其保存在父类中:super(k),value才保存在entry的Object value属性中
这就有一个内存泄漏的问题额:
我们最开始是自己new了一个threadlocal,这个时候是一个强引用,后面如果threadlocal为null,但还是不能被gc回收(如果不使用弱引用),造成内存泄漏;
如果使用了弱引用,那entry中的super(k)传入的threadlocal也为null,导致entry中的value找不到了,也会造成内存泄漏,所以我们一般会在finally{}中加上threadlocal.remove删除对应的entry,即弱引用指向的value