对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()方法

相关推荐
.豆鲨包6 分钟前
【Android】Lottie - 实现炫酷的Android导航栏动画
android·java
donotshow19 分钟前
Spring Boot 整合 ShedLock 处理定时任务重复
java·后端
不够优雅30 分钟前
【Concept Plugin 3】轻量级插件化解决方案|动态类加载
java·spring boot·spring cloud
程序员三明治1 小时前
选 Redis Stream 还是传统 MQ?队列选型全攻略(适用场景、优缺点与实践建议)
java·redis·后端·缓存·rocketmq·stream·队列
Cosmoshhhyyy4 小时前
《Effective Java》解读第5条:优先考虑依赖注入来引用资源
java
.柒宇.6 小时前
力扣hot100----15.三数之和(java版)
java·数据结构·算法·leetcode
程序员卷卷狗6 小时前
JVM 调优实战:从线上问题复盘到精细化内存治理
java·开发语言·jvm
cj6341181506 小时前
【MySQL】mysqldump使用方法
java·后端
JIngJaneIL6 小时前
停车场管理|停车预约管理|基于Springboot的停车场管理系统设计与实现(源码+数据库+文档)
java·数据库·spring boot·后端·论文·毕设·停车场管理系统
杰克尼7 小时前
二分查找为什么总是写错
java·数据结构·算法