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();
    }
}
相关推荐
We་ct几秒前
LeetCode 64. 最小路径和:动态规划入门实战
开发语言·前端·算法·leetcode·typescript·动态规划
CoderCodingNo2 分钟前
【CSP】CSP-J 2019 江西真题 | 次大值 luogu-P5682 (适合GESP四、五级及以上考生练习)
开发语言·c++·算法
djjdjdjdjjdj14 分钟前
PHP函数如何监控CPU温度传感器_PHP读取核心温度硬件值【详解】
jvm·数据库·python
Hui Baby17 分钟前
java抠图
java·开发语言
m0_6138562918 分钟前
c++怎么把多个变量一次性写入二进制文件_结构体对齐与write【实战】
jvm·数据库·python
gihigo199822 分钟前
30节点系统最优潮流计算(MATLAB实现)
开发语言·matlab
许彰午32 分钟前
# 手写一个迷你Tomcat——三步理解Servlet容器的核心原理
java·servlet·tomcat
꧁细听勿语情꧂42 分钟前
向下调整算法,top - k 问题,链式结构二叉树,前中后序遍历
c语言·开发语言·数据结构·算法
qq_460978401 小时前
html标签怎么表示小字号文字_small标签语义说明【操作】
jvm·数据库·python
qq_413502021 小时前
SQL更新语句性能调优技巧_避免对索引列执行函数操作
jvm·数据库·python