ThreadLocal 是为每个线程提供独立副本的变量工具,其值存储在各线程的 threadLocals(ThreadLocalMap)中,key 为弱引用,value 为强引用。

ThreadLocal 不是“线程本地的变量”,而是“为每个线程提供独立副本的变量工具”。它不把变量绑在 ThreadLocal 实例上,而是把变量副本存在每个 Thread 对象内部的 threadLocals(一个 ThreadLocalMap) 里。
Java 中每个 Thread 实例都持有一个私有字段:
ThreadLocal.ThreadLocalMap threadLocals = null;
这个 map 就是线程专属的“储物抽屉”。当你调用 threadLocal.set("abc") 时,实际发生的是:
立即学习“Java免费学习笔记(深入)”;
Thread.currentThread()
threadLocals(若为 null 则新建一个)threadLocal 实例为 key,以 "abc" 为 value,存进该 map所以,不同线程调用同一个 threadLocal.get(),查的是各自线程的 map,自然互不干扰。
ThreadLocalMap 的 Entry 定义是:
static class Entry extends WeakReference
也就是说:key(ThreadLocal 实例)是弱引用,value(你存的数据)是强引用。
这带来一个关键风险:如果 ThreadLocal 实例被回收(比如 static 引用被置 null),但线程还在运行(如线程池中的线程),那么 map 中的 key 变成 null,但 value 还留着——Entry 成为“僵尸条目”,造成内存泄漏。
因此,务必在使用完后调用 threadLocal.remove(),尤其在线程复用场景(如 Web 容器、线程池)中。
所有核心方法逻辑统一:
set(T value):找当前线程 → 初始化或获取 threadLocals → 插入/覆盖 (this, value)get():找当前线程 → 获取 threadLocals → 查 this 对应的 value;若没值,触发 initialValue()(只执行一次)并写入remove():从 threadLocals 中删除以 this 为 key 的 Entry,清掉 value 引用注意:ThreadLocal.withInitial(() -> ...) 是 Java 8 推荐写法,本质就是帮你封装了 initialValue(),更简洁安全。
常见用法包括:
必须警惕的坑:
remove() → 上一个请求的用户信息/traceId 泄露给下一个请求(线程污染)initialValue() 却没声明为 static → 持有外部类引用,放大内存泄漏基本上就这些。原理不复杂,但容易忽略细节。
以上就是Java里ThreadLocal如何工作_Java ThreadLocal原理与使用说明的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号