对ThreadLocal的一些理解

通常情况下,我们创建的变量是可以被任何一个线程访问并修改的。
如果想实现每一个线程都有自己的专属本地变量该如何解决呢?

JDK 中自带的ThreadLocal类正是为了解决这样的问题。
ThreadLocal类主要解决的就是让每个线程绑定自己的值


ThreadLocal与ThreadLocalMap的关系

java 复制代码
public class Thread implements Runnable {
    //......
    //与此线程有关的ThreadLocal值。由ThreadLocal类维护
    ThreadLocal.ThreadLocalMap threadLocals = null;

    //与此线程有关的InheritableThreadLocal值。由InheritableThreadLocal类维护
    ThreadLocal.ThreadLocalMap inheritableThreadLocals = null;
    //......
}

ThreadLocalMap是ThreadLocal类的静态内部类,即一个对象,且ThreadLocalMap中存放的是Entry<K ,V>数组,每一个entry中存放的key为当前ThreadLocal对象,value为我set的值。

因为是静态内部类,所以多个ThreadLocal对象共用一个ThreadLocalMap。

一个Thread线程对象中只有一个ThreadLocalMap

【通过当前线程ID唯一对应一个ThreadLocalMap对象】

但是一个线程可以有多个ThreadLocal对象,当然只是可以....


ThreadLocal如何避免内存泄漏

什么是内存泄漏?

内存溢出:程序中的内存不够使用者继续分配。

内存泄漏:程序中动态分配的内存无法被释放(GC清理),长时间最终导致内存溢出。

这里涉及到弱引用,即entry对象中key对threadlocal对象的引用。

如果 ThreadLocal对象 没有被外部强引用的情况下,在垃圾回收的时候,key 会被清理掉(弱引用),而 value 不会被清理掉。【key是弱引用和value本身是强引用没有关系】

这样一来,ThreadLocalMap 中就会出现 key 为 null 的 Entry。假如我们不做任何措施的话,value 永远无法被 GC 回收,这个时候就可能会产生内存泄露

ThreadLocalMap 实现中已经考虑了这种情况,在调用 set()get()remove() 方法的时候,会清理掉 key 为 null 的记录。

remove核心为expungeStaleEntry(),清除完当前key为null的entry对象,然后还会遍历其他Entry,清空所有key为null的value和 entry作为一个兜底。

使用完 ThreadLocal方法后最好手动调用remove()方法

相关推荐
白起那么早4 分钟前
我又开发了一款idea插件-ContiNewGenerator
java·后端
装不满的克莱因瓶16 分钟前
【Java架构师体系课 | MySQL篇】③ Explain执行计划详解
java·数据库·mysql·架构·优化·索引·explain
王煜苏18 分钟前
最新版idea2025 配置docker 打包spring-boot项目到生产服务器全流程,含期间遇到的坑
java·docker·容器
李玮豪Jimmy30 分钟前
Day18:二叉树part8(669.修剪二叉搜索树、108.将有序数组转换为二叉搜索树、538.把二叉搜索树转换为累加树)
java·服务器·算法
后端小张37 分钟前
【AI 学习】AI Agent 开发进阶:架构、规划、记忆与工具编排
java·人工智能·ai·架构·系统架构·agent·智能体
西岭千秋雪_1 小时前
Kafka客户端整合
java·spring boot·分布式·kafka·linq
leonardee1 小时前
Golang笔记——Interface类型
java·后端
我是好小孩1 小时前
【Android】RecyclerView的高度问题、VH复用概念、多子项的实现;
android·java·网络
张彦峰ZYF1 小时前
高并发优惠权益聚合接口的优雅实现(含超时控制 + 来源标识 + Fallback 降级)
java·后端·面试
4Forsee1 小时前
【Android】模板化解决复杂场景的滑动冲突问题
android·java·rpc