ThreadLocal` 的工作原理

news2025/1/9 11:58:25

ThreadLocal 的工作原理:

ThreadLocal 是 Java 提供的一个类,它用于为每个线程提供独立的变量副本。也就是说,多个线程访问同一个 ThreadLocal 变量时,每个线程看到的值都是不同的,相互隔离,互不干扰。

ThreadLocal 的工作原理是:

  • 每个线程都有一个 ThreadLocalMap(该 Map 存储了线程对应的 ThreadLocal 变量副本)。
  • ThreadLocal 内部维护了一个 ThreadLocalMap,其中 ThreadLocal 作为键,线程局部变量的值作为值。

例如:

ThreadLocal<Integer> threadLocal = ThreadLocal.withInitial(() -> 0);
threadLocal.set(42);

在这个例子中,threadLocal 存储的是每个线程的整数副本,每个线程会持有自己独立的 42 这个值。

为什么 ThreadLocal 会导致内存泄漏?

1. ThreadLocal 的值不会自动清除:
ThreadLocal 的一个重要特点是,它的值是与当前线程相关联的。当线程结束时,理论上 ThreadLocal 中存储的值应该被回收。但实际上,ThreadLocalMapThread 对象的一个字段,并且它的条目(ThreadLocal 和其值)在 ThreadLocalMap 中是通过强引用持有的。

2. 线程池中的线程复用:
在多线程应用程序中(特别是使用线程池的应用程序),线程是被复用的。线程池中的线程会一直存在并且不断地被分配到不同的任务上。如果使用了 ThreadLocal,每次线程复用时,ThreadLocalMap 中的键值对(即线程的局部变量副本)依然存在,直到线程结束或者 JVM 回收线程。这意味着,如果没有显式地清除 ThreadLocal 中的值,这些值将会一直占用内存。

可能导致的内存泄漏场景:

  1. 线程池中未清理的 ThreadLocal 值:
    线程池中的线程是长时间存在的,线程在执行完一个任务后可能会继续用于其他任务。如果在任务执行过程中通过 ThreadLocal 存储了一些对象的引用,而这些对象不再需要时没有显式清理,线程中的 ThreadLocalMap 就会持有这些对象的引用。由于线程池的线程复用,ThreadLocalMap 中的值可能会一直存在,导致内存泄漏。

  2. 没有清理的 ThreadLocal 值:
    ThreadLocal 对象本身不会自动清除,因此在一些场景下,如果 ThreadLocal 对象没有手动清除(例如调用 ThreadLocal.remove()),它所引用的对象可能会一直存在,无法被垃圾回收。

如何避免 ThreadLocal 引起的内存泄漏?

  1. 手动调用 ThreadLocal.remove()
    每次使用 ThreadLocal 后,特别是在使用线程池时,应该显式地调用 ThreadLocal.remove() 来清除存储的值,从而避免线程池线程复用时发生内存泄漏。

    例如:

    ThreadLocal<Integer> threadLocal = ThreadLocal.withInitial(() -> 0);
    
    try {
        threadLocal.set(42);
        // 执行任务
    } finally {
        // 清除 ThreadLocal 值,避免内存泄漏
        threadLocal.remove();
    }
    
  2. 使用 ThreadLocal 的生命周期与线程的生命周期一致:
    确保 ThreadLocal 的使用场景与线程的生命周期一致,避免 ThreadLocal 存储不再需要的对象。当任务执行完成后,及时清理 ThreadLocal

  3. 使用弱引用 WeakReference 或其他策略:
    在某些情况下,可能可以使用 WeakReference 来代替 ThreadLocal 来避免强引用导致的内存泄漏。这样,如果没有强引用到线程局部变量,它们就可以被垃圾回收。

  4. 考虑使用 InheritableThreadLocal
    如果是父线程和子线程之间的传递数据,可以使用 InheritableThreadLocal。但即使使用 InheritableThreadLocal,在合适的时机清理值依然很重要。

总结:

ThreadLocal 在正确使用时非常有用,特别是在需要每个线程存储独立数据的场景中。然而,如果使用不当,尤其是在多线程环境下(例如线程池中),ThreadLocal 可能会导致内存泄漏,特别是当线程池中的线程被复用且没有清理 ThreadLocal 中的值时。为了避免内存泄漏,使用 ThreadLocal 时应当小心,确保在不需要的情况下及时调用 remove() 清理值。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2273773.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

js状态模式

允许一个对象在其内部状态改变时改变它的行为。 状态模式将对象的状态封装成独立的类&#xff0c;并使它们可以互相转换 // 定义状态接口class State {constructor() {if (this.constructor State) {throw new Error(不能实例化抽象类);}}// 定义状态方法handle(context) {th…

平面坐标转大地坐标(arcgisPro中进行)

1、将需要转换的红线导入arcgisPro中&#xff0c;如下&#xff1a; 2、在地图菜单栏中&#xff0c;选择坐标转换工具&#xff0c;如下&#xff1a; 3、打开坐标转换工具 4、开启捕捉 5、 设置大地坐标显示格式 6、如下&#xff1a; 7、显示如图&#xff1a; 8、再依次添加几个待…

(长期更新)《零基础入门 ArcGIS(ArcScene) 》实验七----城市三维建模与分析(超超超详细!!!)

城市三维建模与分析 三维城市模型已经成为一种非常普遍的地理空间数据资源,成为城市的必需品,对城市能化管理至关重要。语义信息丰富的三维城市模型可以有效实现不同领域数据与IS相信息的高层次集成及互操作,从而在城市规划、环境模拟、应急响应和辅助决策等众多领域公挥作用、…

SpringBootWeb 登录认证(day12)

登录功能 基本信息 请求参数 参数格式&#xff1a;application/json 请求数据样例&#xff1a; 响应数据 参数格式&#xff1a;application/json 响应数据样例&#xff1a; Slf4j RestController public class LoginController {Autowiredpriva…

夯实前端基础之HTML篇

知识点概览 HTML部分 1. DOM和BOM有什么区别&#xff1f; DOM&#xff08;Document Object Model&#xff09; 当网页被加载时&#xff0c;浏览器会创建页面的对象文档模型&#xff0c;HTML DOM 模型被结构化为对象树 用途&#xff1a; 主要用于网页内容的动态修改和交互&…

UI自动化测试保姆级教程--pytest详解(精简易懂)

欢迎来到啊妮莫的学习小屋 别让过去的悲伤&#xff0c;毁掉当下的快乐一《借东西的小人阿莉埃蒂》 简介 pytest是一个用于Python的测试框架, 支持简单的单元测试和复杂的功能测试. 和Python自带的UnitTest框架类似, 但是相比于UnitTest更加简洁, 效率更高. 特点 非常容易上手…

有序数据中插入不确定数据保证数据插入的位置顺序正确排序

解决有序数据中插入不确定数据保证数据插入的位置顺序正确排序 前言 java 数据库中存储自增id 有序的数据&#xff0c; 前端页面基于 id 5和 6 之间新增一条数据&#xff0c;在 id 6 和 7之间新增 2条&#xff0c;或者更复杂的场景&#xff0c;后台接口如何保存数据使得页面数…

基于 Apache Commons Pool 实现的 gRPC 连接池管理类 GrpcChannelPool 性能分析与优化

基于 Apache Commons Pool 实现的 gRPC 连接池管理类 GrpcChannelPool 性能分析与优化 1. 输出关键信息的代码示例 日志记录方法 使用以下代码记录连接池的关键信息&#xff0c;帮助分析连接池的状态和性能瓶颈&#xff1a; import org.apache.commons.pool2.impl.GenericO…

不同方式获取音频时长 - python 实现

DataBall 助力快速掌握数据集的信息和使用方式&#xff0c;会员享有 百种数据集&#xff0c;持续增加中。 需要更多数据资源和技术解决方案&#xff0c;知识星球&#xff1a; “DataBall - X 数据球(free)” -------------------------------------------------------------…

在 C# 中显示动画 GIF 并在运行时更改它们

您可以通过将按钮、图片框、标签或其他控件的Image属性设置为 GIF 文件 来显示动画 GIF 。&#xff08;如果您在窗体的BackgroundImage属性中显示一个&#xff0c;则不会获得动画。&#xff09; 有几种方法可以在运行时更改 GIF。 首先&#xff0c;您可以将 GIF 添加为资源。…

Element-plus、Element-ui之Tree 树形控件回显Bug问题。

需求&#xff1a;提交时&#xff0c;需要把选中状态和半选中状态 的数据id提交。如图所示&#xff1a; 数据回显时&#xff0c;会出现代码如下&#xff1a; <template><el-tree ref"treeRef" :data"data" show-checkbox node-key"id" …

【江协STM32】9-1/2/3 USART串口协议、USART外设、串口发送串口发送+接收

1. 通信接口 通信的目的&#xff1a;将一个设备的数据传送到另一个设备&#xff0c;扩展硬件系统通信协议&#xff1a;制定通信的规则&#xff0c;通信双方按照协议规则进行数据收发全双工&#xff1a;指通信双方能够同时进行双向通信。发送线路和接收线路互不影响&#xff0c…

小程序租赁系统开发的优势与应用前景分析

内容概要 小程序租赁系统是一种新兴的数字化解决方案&#xff0c;旨在为用户提供更加便捷与高效的租赁服务。它通常包括一系列功能&#xff0c;如在线浏览、即时预定、支付功能以及用户反馈机制。这些系统在使用上极为友好&#xff0c;让用户能够轻松选择所需的商品或服务&…

25/1/8 算法笔记<强化学习> GYM环境

前几天花了好多时间在装各个仿真环境上&#xff0c;有V-rep,Pybullet,unity的Ml-agent,很多一大堆&#xff0c;好多好多问题差点逼疯我&#xff0c;可能就pybullet能玩一点&#xff0c;到之后学了机器人我再来玩它&#xff0c;最后的最后&#xff0c;我发现还得是我的gym&#…

学习随记:word2vec中归一化处理的作用

答案来自ai&#xff0c;直接复用为参考&#xff1a; 向量归一化的好处 将向量进行归一化&#xff0c;使其模长为 1&#xff08;即投射到单位圆/单位球上&#xff09;&#xff0c;在许多情况下具有实际意义和计算优势。以下是归一化的主要好处和原因&#xff1a; 1. 提高数值稳…

【C++】B2108 图像模糊处理

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C 文章目录 &#x1f4af;前言&#x1f4af;题目描述题目内容输入格式输出格式示例输入&#xff1a;输出&#xff1a; &#x1f4af;题目分析问题拆解 &#x1f4af;我的做法代码实现代码分析 &#x1f4af;老师的做法…

selenium+pyqt5自动化工具总结

说明&#xff1a;本工具是&#xff0c;操作外部google浏览器、selenium是无法操作qt界面中嵌套的浏览器的&#xff0c; 工具在后面 1. 代码结构 pycharm打开的文件下&#xff0c;再写一个子文件&#xff0c;文件导入的时候把子文件名带上 这样就可以在 外层使用命令 pyinst…

经典多模态模型CLIP - 直观且详尽的解释

对比语言-图像预训练&#xff08;CLIP&#xff09;&#xff0c;这是一种创新的多模态建模策略&#xff0c;能够创建视觉和语言的联合表示。CLIP 的效果非常出色&#xff0c;可以用于构建高度特定且性能卓越的分类器&#xff0c;而无需任何训练数据。本文将深入探讨其理论基础&a…

新时期下k8s 网络插件calico 安装

1、k8s master节点初始化完毕以后一直处于notreadey状态&#xff0c;一直怀疑是安装有问题或者是初始化有问题&#xff08;当然&#xff0c;如果真有问题要先解决这些问题&#xff09;&#xff0c;经过不断探索才发现是网络插件没有安装导致的&#xff0c;根据建议安装calico插…

【图像加密解密】Logistic混沌映射的彩色图像加密算法复现(含相关性检验)【Matlab完整源码 1期】

1、说明 本文给出详细完整代码、完整的实验报告和PPT。 环境&#xff1a;MATLAB2019a 复现文献&#xff1a;[1]黄硕.基于改进的Logistic混沌映射彩色图像加密算法[J].河南工程学院学报(自然科学版),2015,27(02):63-67. 主要目的是为了快速了解何为混沌序列、混沌序列产生、…