ThreadLocal的作用和底层原理

作用:

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();
    }
}
相关推荐
沐苏瑶2 小时前
Java数据结构-LinkedList与链表
java·数据结构·链表
ren049182 小时前
JVM复习
jvm
今天你TLE了吗2 小时前
JVM学习笔记:第九章——StringTable字符串常量池
java·jvm·笔记·后端·学习
心前阳光2 小时前
Mirror网络库插件使用4
java·linux·网络·unity·c#·游戏引擎
Rsun045512 小时前
定时任务如何保证任务的可靠性和幂等性?
java
西野.xuan2 小时前
【effective c++】条款四十三:学习处理模版化基类内的名称
java·c++·学习
Nontee2 小时前
Java 后端开发面试技能清单
java·面试
1104.北光c°2 小时前
JVM虚拟机【八股篇】:类加载机制与性能调优
java·开发语言·jvm·笔记·程序人生·调优·双亲委派