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();
    }
}
相关推荐
青山木几秒前
Hot 100 --- 轮转数组
java·数据结构·算法
凡人叶枫29 分钟前
Effective C++ 条款22:将成员变量声明为 private
linux·开发语言·c++
Qt程序员35 分钟前
掌握 Linux 内核调度:从原理到实现(进程篇)
java·开发语言
code bean40 分钟前
【LangChain】检索器完全指南:从向量检索到生产级 RAG 架构
java·开发语言·微服务
大白菜和MySQL41 分钟前
java应用排查高线程
java·python
KobeSacre1 小时前
ReentrantLock源码
java
LabVIEW开发1 小时前
LabVIEW + MATLAB 混合编程:爆炸场测试数据精准采集方案
开发语言·matlab·labview
嵌入式协会20240721 小时前
(已解决)MinIO python 获取预签名出现forbidden、errornetwork等错误
java·开发语言·python
宸丶一1 小时前
Day 14:任务追踪 - 让 Agent 拥有项目管理能力
开发语言·python
小短腿的代码世界1 小时前
Qt行情协议解析与二进制编解码优化:从FIX到自定义协议的全链路架构
开发语言·qt·架构