ThreadLocal 详解

1. 是什么

ThreadLocal 是线程本地变量,给每个线程提供独立的变量副本,互不影响。

2. 简单用法

java 复制代码
public class UserContext {
    private static final ThreadLocal<String> USER = new ThreadLocal<>();

    public static void set(String user) {
        USER.set(user);
    }

    public static String get() {
        return USER.get();
    }

    public static void remove() {
        USER.remove();
    }
}

// 使用
UserContext.set("张三");
String user = UserContext.get();  // 同一个线程里拿到的都是"张三"
UserContext.remove();

3. 常见场景

用户信息传递

不用每次方法参数传递:

java 复制代码
// 登录时存进去
UserContext.set(SecurityContext.getUser());

// 任何地方都能取到
public void anyMethod() {
    User user = UserContext.get();  // 拿到当前登录用户
}

数据库连接

保证同一线程用的是同一个连接:

java 复制代码
private static final ThreadLocal<Connection> DB = new ThreadLocal<>();

// 获取连接
public Connection getConnection() {
    Connection conn = DB.get();
    if (conn == null) {
        conn = dataSource.getConnection();
        DB.set(conn);
    }
    return conn;
}

请求Tracer

java 复制代码
private static final ThreadLocal<String> TRACE_ID = new ThreadLocal<>();

public void handleRequest() {
    TRACE_ID.set(UUID.randomUUID().toString());
    log.info("开始处理, traceId={}", TRACE_ID.get());
    // 整个调用链都能拿到同一个 traceId
}

4. 原理

每个 Thread 对象里有个 ThreadLocalMap:

复制代码
Thread
  └── threadLocals: ThreadLocalMap
        ├── ThreadLocal<?>  →  value
        ├── ThreadLocal<?>  →  value
        └── ...

set 时,把 ThreadLocal 对象作为 key,存到当前线程的 Map 里。

get 时,从当前线程的 Map 里,用 ThreadLocal 对象作为 key 取值。

所以不同线程访问同一个 ThreadLocal,拿到的值不一样。

5. 内存泄漏

ThreadLocalMap 的 Entry 继承 WeakReference:

java 复制代码
static class Entry extends WeakReference<ThreadLocal<?>> {
    Object value;
    Entry(ThreadLocal<?> k, Object v) {
        super(k);
        value = v;
    }
}

key(ThreadLocal 对象)是弱引用,理论上会被 GC 回收。但 value 是强引用,可能造成泄漏。

所以用完要 remove!

java 复制代码
try {
    User user = UserContext.get();
} finally {
    UserContext.remove();  // 用完清理
}

6. 总结

  • 每个线程独立副本,互不干扰
  • 常用场景:用户信息传递、数据库连接、链路追踪
  • 用完记得 remove,防止内存泄漏
相关推荐
薛定谔的悦1 小时前
光伏-储能-负荷联合预测:给 EMS 装上“预知能力“
java·数据库·人工智能·python·储能
大菜菜小个子1 小时前
template<typename T>使用
java·开发语言·算法
Refrain_zc1 小时前
Android开发: 拒绝 Activity 重建!onConfigurationChanged 实现平板横竖屏无缝切换
java
方也_arkling2 小时前
【Java-Day15】API篇-ArrayList集合
java·开发语言
AI人工智能+电脑小能手2 小时前
【大白话说Java面试题 第89题】【Mysql篇】第19题:Hash 索引和 B+ 树索引的区别?它们在使用方面的区别?
java·数据库·mysql·面试·哈希算法
我是一颗柠檬2 小时前
【Java后端技术亮点】动态路由权限(按钮级权限),细粒度控制到按钮级别
java·开发语言·后端·状态模式
Fanfanaas2 小时前
C++ 继承
java·开发语言·jvm·c++·学习·算法
蚰蜒螟2 小时前
走进 Linux 内核:从 touch 命令到磁盘 inode 的完整旅程
java·linux·前端
zzqssliu2 小时前
taocarts 跨境独立站 SEO 优化实践(多语言 + 反向海淘场景)
java·javascript·php
在繁华处2 小时前
Java从零到熟练(十一):Spring框架入门
java·开发语言·spring