ThreadLocal提供了线程本地变量,当创建一个ThreadLocal变量后,每个线程访问ThreadLocal变量的时候访问的都是自己线程中的变量,而不是共享变量。
1 ThreadLocal类的使用
先看一个例子
public class TestThreadLocals {
static ThreadLocal<String> local = new ThreadLocal<>();
public static void main(String[] args) {
new Thread(new Runnable() {
@Override
public void run() {
// 线程1中设置ThreadLocal变量
local.set("thread-1 local variable");
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println("thread-1: " + local.get());
}
}).start();
new Thread(new Runnable() {
@Override
public void run() {
//local.set("thread-2 local variable");
System.out.println("thread-2: " + local.get());
}
}).start();
}
}
线程1先将local变量设置为“thread-1 local variable” ,让出CPU,线程2中获取local变量
输出结果: 可见线程2中local变量并没有改变
将注释行//local.set("thread-2 local variable");的注释去掉后,输出结果,两个线程内设置local互不影响
2 ThreadLocal的实现原理
ThreadLocal的类图
Thread类中有两个成员变量 threadLocals和 inheritableThreadLocals, 它们都是ThreadLocalMap类型的,ThreadLocalMap是ThreadLocal的一个静态内部类,作用和hashmap差不多,用于存放某个线程中的ThreadLocal变量以及对应的变量值,因此可以说明一个线程可以具有多个ThreadLocal变量。
2.1 ThreadLocal变量值是如何被存放到线程中的
set(T)方法
首先获取当前线程,通过getMap方法获取Thread对象中的threadLocals映射表,如果threadLocals存在,将ThreadLocal对象作为key,对应的值作为value存入该映射表中。否则,创建线程的threadLocals变量。
由此可见,ThreadLocal只是一个工具壳,它通过操作Thread的成员变量将value的值存入线程。
T get()方法
get方法基本同理,通过当前线程获取线程的threadLocals映射表,从映射表中查找是否存在key = this的项,有则返回,没有的话就返回一个空值
3 ThreadLocal可能引起内存泄露
每个线程的本地变量存放在线程自己的内存变量 threadLocals 中,如果当前线程一直不消亡 , 那么这些本地变量会一直存在 , 所以可能会造成内存溢出 , 因此使用完毕后要记得调用 ThreadLocal 的 remove 方法删 除对应线程的 threadLocals 中的本地变量