大家好,这里是淇妙小屋,一个分享技术,分享生活的博主0. 类图和get()方法调用图
以下是我的主页,各个主页同步更新优质博客,创作不易,还请大家点波关注
掘金主页
知乎主页
Segmentfault主页
开源中国主页
后续会发布更多MySQL,Redis,并发,JVM,分布式等面试热点知识,以及Java学习路线,面试重点,职业规划,面经等相关博客
转载请标明出处!
文章图片
1. 类关系
- 每个Thread都有一个ThreadLocalMap,ThreadLocalMap中存储着多个Entry对象,这些Entry对象就是该线程独有的变量副本
- 每个Entry都可以绑定一个ThreadLocal实例(ThreadLocal是key,Entry是value)
- ThreadLocalMap中的 Entry[ ]table可以理解为一个键值对,key为ThreadLocal实例,value为Entry对象(表示线程在这个ThreadLocal实例上绑定的变量副本),table采用开放地址法解决冲突
文章图片
有以下三种情况
- 线程的ThreadLocalMap未初始化,为空
绿色执行路径
- 线程的ThreadLocalMap已初始化,但是该ThreadLocal实例对应的索引上不存在Entry
蓝色执行路径
- 线程的ThreadLocalMap已初始化,并且该ThreadLocal实例对应的索引上存在Entry
红色执行路径
基本逻辑同get()
4. ThreadLocal内存泄漏和Entry清除
4.1 弱引用避免ThreadLocal内存泄漏 Entry继承自WeakReference,注意Entry不是弱引用,Entry中的reference才是弱引用,这个弱引用避免了ThreadLocal的内存泄漏
文章图片
我们一般使用线程池,线程的生命周期很长,所以Thread引用会存在很长一段时间
如果我们不使用ThreadLocal了,将ThreadLocal引用设置为null(或者方法执行完毕,ThreadLocal引用弹出栈)
文章图片
没有了ThreadLocal引用,ThreadLocal无法被访问到,线程就无法使用ThreadLocal,此时GC应该要回收掉ThreadLocal对象
若Entry的引用是强引用,会导致ThreadLocal无法被回收,造成内存泄漏
所以Entry的引用必须是弱引用,这样一旦发生GC,ThreadLocal就会被回收
4.2 Entry清除 ThreadLocal一旦被回收,那么Entry对象的reference就为空,此时Entry也是无用的,应该被回收掉——expungeStaleEntry()会回收掉那么reference为null的Entry对象(详情看源代码或者0. 类图和get()方法调用图)
【Java并发—ThreadLocal】许多个地方都会调用expungeStaleEntry()来清除无用的Entry,例如get()和扩容
推荐阅读
- Java并发—Java线程
- Java并发—sychronized关键字
- Java并发—volatile关键字
- Spring Boot 最新发布!Java 18 都加入了!!学不动了。。
- 技术分享|MRO工业品行业产业转型,建立B2B集采平台发展业态模式
- 资讯|从苹果 M1 到英伟达 Grace,“缝合风”为何在芯片大厂中盛行()
- java|2020年网络安全状况透视
- Java安全|Spring Cloud Gateway_CVE-2022-22947漏洞复现
- java|Java如何使用实时流式计算处理()