ThreadLocal简介
ThreadLocal是一个线程内部的数据存储类,它可以为每个线程提供独立的变量副本,不同线程间的变量无法相互访问和修改。这避免了每个线程都要维护一套独立变量的麻烦,并且也减少了线程之间不必要的数据争用。ThreadLocal适用于这样的场景:每个线程需要有自己单独的实例,而不是共享实例。例如,在 web 应用中,每个请求被一个新的线程处理,每个线程需要有自己的变量实例。
ThreadLocal使用示例
public class ThreadLocalExample {
// 线程局部变量,每个线程有自己的变量副本
private ThreadLocal<String> threadLocal = new ThreadLocal<>();
public void set(String value) {
threadLocal.set(value);
}
public String get() {
return threadLocal.get();
}
}
public class ThreadLocalTest {
public static void main(String[] args) {
ThreadLocalExample example = new ThreadLocalExample();
// 线程1设置threadLocal变量
example.set("Thread1 local variable");
System.out.println("Thread1 get: " + example.get());
// 线程2无法获取线程1设置的threadLocal变量
Thread thread2 = new Thread() {
public void run() {
example.set("Thread2 local variable");
System.out.println("Thread2 get: " + example.get());
}
};
thread2.start();
}
}
复制代码
运行结果: Thread1 get: Thread1 local variable Thread2 get: Thread2 local variable每个线程获取自己设置的值,并不同线程间互不干扰。
ThreadLocal原理解析
ThreadLocal内部使用ThreadLocalMap来存储每个线程的变量副本。ThreadLocalMap是ThreadLocal的静态内部类,每个线程都有自己的ThreadLocalMap副本。
ThreadLocal中get()方法的实现如下:
public T get() {
Thread t = Thread.currentThread();
ThreadLocalMap map = getMap(t);
if (map != null) {
ThreadLocalMap.Entry e = map.getEntry(this);
if (e != null) {
@SuppressWarnings("unchecked")
T result = (T)e.value;
return result;
}
}
return setInitialValue();
}
复制代码
- 获取当前线程对象
- 获取当前线程的ThreadLocalMap(实际上是从当前线程的ThreadLocalMap变量中获取)
- 在ThreadLocalMap中获取当前ThreadLocal变量对应的value值
- 如果不存在,调用setInitialValue()方法初始化value值,并存储到Th