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();
    }
}
相关推荐
二哈赛车手7 小时前
新人笔记---ApiFox的一些常见使用出错
java·笔记·spring
为何创造硅基生物7 小时前
C语言 结构体内存对齐规则(通俗易懂版)
c语言·开发语言
吃好睡好便好8 小时前
在Matlab中绘制横直方图
开发语言·学习·算法·matlab
栗子~~8 小时前
JAVA - 二层缓存设计(本地缓冲+redis缓冲+广播所有本地缓冲失效) demo
java·redis·缓存
星寂樱易李8 小时前
iperf3 + Python-- 网络带宽、网速、网络稳定性
开发语言·网络·python
YDS8298 小时前
DeepSeek RAG&MCP + Agent智能体项目 —— RAG知识库的搭建和接口实现
java·ai·springboot·agent·rag·deepseek
仰泳之鹅8 小时前
【C语言】自定义数据类型2——联合体与枚举
c语言·开发语言·算法
之歆8 小时前
DAY_12JavaScript DOM 完全指南(二):实战与性能篇
开发语言·前端·javascript·ecmascript
未若君雅裁9 小时前
MyBatis 一级缓存、二级缓存与清理机制
java·缓存·mybatis
cen__y9 小时前
Linux12(Git01)
linux·运维·服务器·c语言·开发语言·git