作用:
ThreadLocal使得每个线程都可以创建自己共享变量的副本,避免了多线程竞争共享变量
底层原理
当创建ThreadLocal变量时,每个访问该变量的线程都会拥有一个独立的副本,线程通过get( )方法会读取自己线程的副本,set( )时向自己线程的副本写数据。从而解决了线程安全的问题。
public class ThreadLocalExample {
private static ThreadLocal<Integer> threadLocal = ThreadLocal.withInitial(() -> 0);
public static void main(String[] args) {
Runnable task = () -> {
int value = threadLocal.get();
value += 1;
threadLocal.set(value);
System.out.println(Thread.currentThread().getName() + " Value: " + threadLocal.get());
};
Thread thread1 = new Thread(task, "Thread-1");
Thread thread2 = new Thread(task, "Thread-2");
thread1.start(); // 输出: Thread-1 Value: 1
thread2.start(); // 输出: Thread-2 Value: 1
}
}
Thread、ThreadLocal和ThreadLocalMap的关系
答:一个程序只有可以创建多了ThreadLocal对象,每个Thread实例可以通过ThreadLocal对象的set方法向自己的ThreadLocalMap中记录对应的值
每个Thread类中有ThreadLocalMap成员变量。
ThreadLocal在程序中创建,相当于一个存储物体的标签
举个例子:
因为一个程序可以创建多个 ThreadLocal****对象,这就意味着在实际开发中,你可能会有:
ThreadLocal<User> userLocal(标签1:存当前登录用户信息)ThreadLocal<Connection> dbLocal(标签2:存数据库连接)ThreadLocal<String> traceIdLocal(标签3:存日志追踪 ID)
当"线程A"在执行任务时,它通过这三个 ThreadLocal 分别调用了 set() 方法。此时,线程A的 ThreadLocalMap**,就整整齐齐地放了三个键值对**。
- Key是
userLocal,Value 是用户信息;可以通过userLocal.get()方法来获取当前线程存储的登录用户信息 - Key是
dbLocal,Value 是数据库连接;

可能存在内存泄漏的问题
ThreadLocalMap中key存储的是ThreadLocal变量,value是当前线程要存储的值。key是弱引用、value是强引用。
如果ThreadLocal变量不再被强引用指向时,垃圾回收器会在下次回收时GC该实例。
结果:ThreadLocalMap中的键值对key变为null,value还是存储真实的值。
解决:在使用完ThreadLocal对象后,务必确保该对象被释放。使用 try-finally 块可以确保即使发生异常,remove() 方法也一定会被执行
// 定义为 static final,避免重复创建 ThreadLocal 实例
private static final ThreadLocal<UserContext> userContextHolder = new ThreadLocal<>();
public void processRequest(HttpServletRequest request) {
try {
// 在 try 块中设置值
UserContext context = buildUserContext(request);
userContextHolder.set(context);
// 执行业务逻辑
doBusinessLogic();
} finally {
// 在 finally 块中必须清理,确保无论是否发生异常都会执行
userContextHolder.remove();
}
}