ThreadLocal就可以看成是一个Map,Map的key是:当前ThreadLocal对象 + 当前线程
ThreadLocal使用方式
public void test1() {
ThreadLocal tl = new ThreadLocal();
tl.set(new Person())
new Thread(()-> { System.out.println(tl.get()) }).start()
tl.remove()
}
void main(){
test1();
}
//上面的打印结果是null
ThreadLocal set一个对象的流程:
首先java的Thread类有一个字段threadLocals
public class Thread {
ThreadLocal.ThreadLocalMap threadLocals = null;
}
public class ThreadLocal {
public void set(T value) {
Thread t = Thread.currentThread();
//首先尝试获取当前线程的threadlocals字段,改字段是ThreadLocalMap类型
ThreadLocalMap map = t.threadLocals
if(map == null) {
map = t.threadLocals = new ThreadLocalMap()
}
//ThreadLocal的key是当前自己对象
map.set(this,value)
}
}
ThreadLocal的引用关系
假如有如下代码:
public void test1() {
ThreadLocal t1 = new ThreadLocal();
t1.set(new Object());
}
public void main(String[] args) {
test1();
System.out.println("finish");
}
上面代码的引用关系如下:
1. 局部变量 t1 指向 ThreadLocal堆内存
2. 当前线程的threadLocals 指向了 ThreadLocalMap
3. ThreadMap 的key 指向了 ThreadLocal, value指向了Object
如果函数test1执行完成之后,t1指向了ThreadLocal的引用会释放,
但是 ThreadMap的key指向的 ThreadLocal对象的引用不会释放,所以这个引用是弱引用,防止内存泄漏