当前位置: 首页 > news >正文

ThreadLocal结构

目录

一、ThreadLocal 要解决的核心问题

二、核心结构

1.Thread 类(线程载体)

2.ThreadLocal 类(访问入口&健生成器)

3.ThreadLocal.ThreadLocalMap 类(线程专属存储结构)

三、核心关系图:一切尽在图中

四、关键设计


ThreadLocal 是 Java 中实现线程隔离数据存储的核心工具。

一、ThreadLocal 要解决的核心问题

提供一种机制,让每个线程都能独立访问自己专属的变量副本,避免多线程环境下的竞争问题。常用于存储线程上下文信息(如用户会话)、数据库连接等。

二、核心结构

1.Thread 类(线程载体)

  • 每个 Thread 对象内部都有一个私有成员变量:threadLocals
  • threadLocals 的类型是 ThreadLocal.ThreadLocalMap
  • 这个 ThreadLocalMap 就是该线程专属的“储物柜”
// java.lang.Thread
ThreadLocal.ThreadLocalMap threadLocals = null;

2.ThreadLocal 类(访问入口&健生成器)

  • 开发者直接操作的类( get()set()remove()
  • 它本身不存储数据,而是作为访问 ThreadLocalMap 的媒介键(key)
  • 每个 ThreadLocal 实例通过其唯一的 threadLocalHashCode(在构造函数中计算生成)标识自己,用于在 ThreadLocalMap 中查找对应的值

3.ThreadLocal.ThreadLocalMap 类(线程专属存储结构)

  • ThreadLocalMap 是 ThreadLocal 的静态内部类
  • 它是线程(Thread)的私有成员变量(threadLocals)
  • 它本质上是一个定制化哈希表,专门用于存储以 ThreadLocal 为键、用户数据为值的键值对
  • 核心结构:Entry[] table
    (1)Entry 是继承自 ThreadLocalMap 的静态内部类
    (2)Entry 继承自 WeakReference<ThreadLocal<?>>
    (3)键(key):是 ThreadLocal 实例的弱引用(WeakReference)
    (4)值(value):是用户存储的实际数据(强引用)
// ThreadLocal.ThreadLocalMap.Entry
static class Entry extends WeakReference<ThreadLocal<?>> {Object value; // 用户存储的实际数据Entry(ThreadLocal<?> k, Object v) {super(k); // 将ThreadLocal作为弱引用value = v;}
}

三、核心关系图:一切尽在图中

ThreadLocal 的数据结构

四、关键设计

首先搞清楚 Java 的四种引用类型

  • 强引用:我们常常 new 出来的对象就是强引用类型,只要强引用存在,垃圾回收器将永远不会回收被引用的对象,哪怕内存不足的时候
  • 软引用:使用 SoftReference 修饰的对象被称为软引用,软引用指向的对象在内存要溢出的时候被回收
  • 弱引用:使用 WeakReference 修饰的对象被称为弱引用,只要发生垃圾回收,若这个对象只被弱引用指向,那么就会被回收
  • 虚引用:虚引用是最弱的引用,在 Java 中使用 PhantomReference 进行定义。虚引用中唯一的作用就是用队列接收对象即将死亡的通知

1.弱引用键(key)

  • 目的:解决 ThreadLocal 对象本身的内存泄漏问题
  • 机制:当 ThreadLocal 实例(tlRef)不再被任何强引用指向时(例如开发者将其设置为 null),仅剩下 Entry.key 这个弱引用。在下一次 GC 发生时,tlRef 就会被回收,对应的 Entry.key 变为 null
  • 价值:防止因为线程长期存在(如线程池线程)导致 ThreadLocal 对象无法回收

2.强引用值(value)

  • 问题:如果 ThreadLocal 被回收(Entry.key == null),但 Entry.key 仍然被 ThreadLocalMap 的 Entry 强引用着,而 ThreadLocalMap 又被线程(Thread)强引用着。如果长期存在(如线程池),这个 value 就会一直占用内存,造成真正的内存泄漏
  • 解决方案:开发者必须显示调用 ThreadLocal.remove() 来删除不再需要的条目,或者在 get() / set() 过程中,ThreadLocalMap 会自动清理 key == null 的陈旧条目(惰性清理)

3.ThreadLocalMap 过期 key 的数据清理方式:探测式清理启发式清理

http://www.lryc.cn/news/588379.html

相关文章:

  • 在Maven多模块项目中进行跨模块的SpringBoot单元测试
  • 考研408《计算机组成原理》复习笔记,第三章(4)——主存与CPU连接(字、位扩展)
  • 研究人员利用提示注入漏洞绕过Meta的Llama防火墙防护
  • 开源软著源代码生成工具(自制)
  • Java行为型模式---模板方法模式
  • 实现高效、可靠的基于骨骼的人体姿态建模(第二章 基于三维人体姿态回归的语义图卷积网络)
  • 如何将 iPhone 备份到云端:完整指南
  • ubuntu系统在线安装postgres
  • 【一维 前缀和+差分】
  • 【牛客刷题】小红的数字删除
  • 第 2 章 数据类型及其运算
  • 内测分发平台应用的异地容灾和负载均衡处理和实现思路
  • 【深度学习笔记】2 浅层神经网络
  • Dubbo 学习笔记
  • python接口自动化 - 使用requests库发送http请求
  • Datawhale AI夏令营——用户新增预测挑战赛
  • Docker入门指南(超详细)
  • 华为OD 消消乐游戏
  • LLaMA.cpp HTTP 服务参数: --pooling 嵌入模型 池化类型详解
  • 【时时三省】(C语言基础)用数组名作函数参数
  • 75、【OS】【Nuttx】【启动】caller-saved 和 callee-saved 示例
  • 数电汇总——logisim的辛酸史
  • 【Python进阶】深度复制——deepcopy
  • stm32-Modbus主机移植程序理解以及实战
  • JSCPC 2025 江苏省赛
  • 制造业实战:数字化集采如何保障千种备件“不断供、不积压”?
  • Java从入门到精通!第五天(面向对象(二))
  • 《解锁音频处理新姿势:探索Librosa的无限可能》
  • HarmonyOS应用无响应(AppFreeze)深度解析:从检测原理到问题定位
  • ISO-IEC-IEEE 42010架构规范