大家好,我是奇喵小屋,一个分享技术和生活的博主。以下是我的主页。各首页同步更新优质博客。创建起来并不容易。还请点开掘金主页了解主页Segmentfault主页开源中国更多MySQL、Redis、并发、JVM、分布式等面试热点知识稍后会在主页发布,还有Java学习路线、面试重点、职业规划、面经等相关博文转载请注明出处!0.类图和get()方法调用图1.类关系每个Thread都有一个ThreadLocalMap,里面存放了多个Entry对象,这些Entry对象是线程唯一的变量副本。每个Entry都可以绑定一个ThreadLocal实例(ThreadLocal是key,Entry是value)ThreadLocalMap中的Entry[]table可以理解为一个键值对,key是ThreadLocal实例,value是Entry对象(表示线程绑定到ThreadLocal实例的变量的副本),该表采用开放地址的方式来解决冲突2、线程在以下三种情况下调用ThreadLocal.get()。线程的ThreadLocalMap没有初始化,为空。绿色执行路径线程的ThreadLocalMap已初始化,但ThreadLocal实例对应的索引不存在实例3、线程调用ThreadLocal.set()的基本逻辑同get()4、ThreadLocal内存泄漏和Entry清除4.1弱引用避免ThreadLocal内存泄漏。Entry继承自WeakReference。请注意,Entry不是弱引用。Entry中的引用是弱引用。这种弱引用避免了ThreadLocal内存泄漏。我们一般使用线程池。线程的生命周期很长,所以Thread的引用会很长一段时间,如果我们不使用ThreadLocal,就把ThreadLocal引用设置为null(或者方法执行完,ThreadLocal引用弹出栈)。没有ThreadLocal引用,就不能访问ThreadLocal,线程也不能使用ThreadLocal。当GC应该回收ThreadLocal对象时,如果Entry的引用是强引用,ThreadLocal无法回收,造成内存泄漏。所以Entry的引用必须是弱引用,这样一旦GC发生,ThreadLocal就会被回收。4.2一旦Entry清空ThreadLocal被回收,则Entry对象的引用为空,此时Entry无用,应该被回收——expungeStaleEntry()会回收引用为null的Entry对象(见源码)codeor0.详见类图和get()方法调用图)很多地方会调用expungeStaleEntry()清除无用的Entry,比如get()和expansion
