快速过一遍ThreadLocal源码
迪丽瓦拉
2025-05-30 10:51:55
0

Thread属性之ThreadLocalMap

ThreadLocal是java的用来做线程隔离的一个类,原意就是一个线程私有变量,用来保证线程的安全。那ThreadLocal是如何实现的呢?我们来看看java的源码,如下Thead类的有一个ThreadLocal的内部类ThreadLocalMap属性

ThreadLocalMap用来表示这个Thread线程拥有的所有ThreadLocal变量,因为每个线程可能实例化多个ThreadLocal变量,所以要用一个Map存储这个线程所有的ThradLocal,然后以ThreadLocal对象引用为key,ThreadLocal要代码的实际值为value。

然后我们在ThreadLocalMap的定义,他虽然命名以Map结尾,但他并没有实现Map接口,他的每一项用一个Entry表示,请看Entry继承了WeakReference类,WeakReference代表弱引用,也就是在jvm内存不足的情况下,GC会清除WeakReference引用的对象所占内存空间,但是重点来了,Entry的构造方法在初始化的时候,是用Entry的key即ThreadLocal做为WeakReference要引用的对象,而没有把Entry本身作为WeakReference要引用的对象,这就导致了ThreadLocal可能会在因为没有硬引用且内存不足的情况下被GC清除,而ThreadLocal的实际代表的值value却没有被GC清除,从而导致内存泄漏

我们再看看ThreadLocal主要的方法

ThreadLocal构造方法

如图所示,无参构造方法什么也不做,只是简单创建一个ThreadLocal实例

但是默认的ThreadLocal对象,在没有用set方法初始化所代表的值的时候,会返回null,如下图源码所示:

如果希望ThreadLocal在没有值的时候调用get()方法自动初始化一个值,可以使用ThreadLocal.withInitial(Supplier)静态方法实例化ThreadLocal类,如下图所示:

withInitial静态方法会返回一个SuppliedThreadLocal实例,如下图我们看到SuppliedThreadLocal是ThreadLocal的子类,也是ThreadLocal的内部静态类,它唯一作用就是重写了ThreadLocal.initialValue()方法,通过调用supplier对象的get()返回一个值

Supplier是一个函数接口,主要用来通过实现get()方法来返回一个值,所以没有可以用类似`ThreadLocal userContext = ThreadLocal.withInitial(() -> new User());`的语句来创建一个拥有默认值的ThreadLocal实例

ThreadLocal之set方法

set方法用于设置ThreadLocal所代表的值,如下图所示,其他ThreadLocal类本身并没有value属性,而是通过获取当前线程的ThreadLocalMap属性,然后以ThreadLocal自己this作为ThreadLocalMap中entry的key,value做ThreadLocalMap中entry的value来保存值,如果Thread的ThreadLocalMap属性为空会通过createMap要创建ThreadLocalMap对象,并发value作为第一个值初始化到ThreadLocalMap中

ThreadLocal之get方法

get方法用于获取ThreadLocal对象代表的实际的值,如下图所示,先获取当前线程,然后获取线程的ThreadLocalMap属性,如果ThreadLocalMap不为空则以当前ThreadLocal的this为key获取Map里面的值,即返回ThreadLocal代表的值,如果ThreadLocalMap为空则调用setInitialValue方法初始化ThreadLocal的值

setInitialValue会首先调用initialValue方法,initialValue方法前面讲过,默认放回null,如果采用`ThreadLocal.withInitial(() -> new User());`的方法会重写initialValue放回一个自定义的值;最后setInitialValue方法会返回initialValue返回的值。

ThreadLocal之remove方法

remove方法用于删除ThreadLocal代表的值,如下图所示,通过当前线程获取线程中ThreadLocalMap,然后通过ThreadLocal的this为key,删除Map中的值。前面已经分析过,用于ThreadLocal后一直要记得调用remove方法删除其代表的值,否则可能导致内存泄漏。因为ThreadLocal可能会被GC回收,但是ThreadLocal代表的value却一直停留在线程的ThreadLocalMap中,一直被线程引用这,如果线程一直没有被销毁(如采用线程池技术中的核心线程,可以重复使用,不会销毁),将会导致内存泄漏

分析完毕

相关内容

热门资讯

处理大数据的最佳方式——con... css属性处理大数据 content-visibility——只需一行CSS代码,让长...
Vuex由浅入深详细讲解 目录前言一,理解Vuex1.1 Vuex是什么1.2 Vuex概述1.3 Vuex统一...
Vector - CAPL -... 摸鱼聊天、答疑解惑首选之地 --- 车载网络哪些事儿你是否还在为VT板卡系统昂贵而发愁?...
qt5.6(mingw) 编译... 具体版本和操纵参考官网win msvc版本编译,Qt5.6编译PCL环境 备注...
【论文速览】引入motion ... 文章目录研究背景解决思路实验效果思考参考资料 收录于 ECCV 2020,代码地址...
字符串 - 二进制和文本字符串... 1.应用场景 主要用于探究字符串中的二进制和文本字符串,以及它们的区别和应用场景。2....
windows下使用gitea... windows下使用gitea搭建git服务器 详细过程 1、简述 使用过好几个git服务器...
LCHub:一句话让 AI 替... LCHub 3 月 18 日消息,这几天,GPT-4 接入微软 Office 全家桶的消息传遍了互联...
STM32模拟SPI控制NRF... STM32模拟SPI控制NRF24L01发送和接收 NRF24L01是一款2.4Ghz ISM频段无...
Vue路由及状态管理 1,Vue引入路由配置 在Vue中,我们可以通过vue-router路...
Lua 开发过程中常见坑 Lua 开发过程中常见坑 Lua next return _G.next( tb ) =&#...
蓝桥c++数位排序(运用pai... 问题描述 小蓝对一个数的数位之和很感兴趣, 今天他要按照数位之和给数排序。当 两个数各个数位之和不同...
3Dmax开孔打洞圆孔洞的几种... 最常见的几种打洞方法及技巧和注意事项总结如下。除了布尔运算外其余几种都是在可编辑多边形模式下进行的&...
04 - 进程参数编程 ---- 整理自狄泰软件唐佐林老师课程 查看所有文章链接:(更新中&...
【Git使用学习】本地创建项目... 记录学习过程,一波三折。以【Vue+CesiumJS学习(1&#...
电子拣货标签1代系统简介 CK_Label_v1 ​一、产品参数 1. 电池供电版 产品型号 CK_Label_v1 尺...
为什么写分页器的时候要用当前页... 比方说总共有100条数据(编号从0起算就是0到99),每次...
古典密码学 主要划分方式及其分类按密钥方式划分:对称,非对称按明文处理方式分...
C# 项目名称为什么是xxx.... 在 C# 中,通常将项目命名为“xxx.xxx”的形式,其中第一个“xx...
【Java】UDP网络编程 文章目录前言DatagramSocketDatagramPacket注意事项与区别代码演示 前言 U...