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

SpringBootWeb 篇-深入了解 ThreadLocal 存在内存泄漏问题

🔥博客主页: 【小扳_-CSDN博客】
❤感谢大家点赞👍收藏⭐评论✍

文章目录

        1.ThreadLocal 概述

        2.ThreadLocal 的主要缺点:

        2.1 存在内存泄漏

        2.2 为什么ThreadLocalMap 的 key 是弱引用?

        2.3 采用线性探测方法

        3. 不使用线程池会出现内存泄漏吗?


        1.ThreadLocal 概述

        ThreadLocal 是每一个线程都独立拥有的,是线程隔离的。且每个线程只能访问自己的 ThreadLocalMap,是一个 key - value 结构,key 代表着 threadLocal 对象,而 value 代表着对应设置的值。因此,每一个线程中只有一个 ThreadLocalMap,ThreadLocalMap 中可以存放着多个 threadLoal 对象和对应着 value 值,即 key - value 结构。

        2.ThreadLocal 的主要缺点:

        2.1 存在内存泄漏

        在线程池技术下,使用 ThreadLocal 会出现内存泄漏问题。因为,当线程执行完毕之后,引用着 threadLocal 对象的成员变量会直接出栈,这时候,对于这个 threadLocal 对象就没有变量对其进行操作了,会一直存在,且没有办法使用,这就是内存泄漏。

        即使当前引用 threadLocal 对象是一个静态变量,可能会一直引用着这个 threadLocal 对象。但是类加载器是可以被卸载的,同样的,这个 threadLocal 对象也会没有变量对其进行操作。

        2.2 为什么ThreadLocalMap 的 key 是弱引用?

        是为了减少内存泄漏。当出现没有栈中没有变量继续引用 threadLocal 对象时,且执行 gc 的时候,因为 key 是弱引用,会被置为 null,在 ThreadLocal 每次 get 或者是 set 的时候会对 Entry 进行一个清空 key 为 null 的操作,当发现出现 key 为 null 的时候,会对 value 也置为 null,同时将这个 key - value 从 threadLocalMap 中进行移除操作。但是这个清除 Entry 操作不是每次从头开始,因此还是会出现内存泄漏的风险。

        避免使用 ThreadLocal 出现内存泄漏,应该在使用完之后,进行 remove 操作!

       2.3 采用线性探测方法

        当 ThreadLocalMap 存储一个 threadLocal 对象的时候,采用的是线性探测方法,当遇到 Hash 冲突的时候,性能不高。提高性能的方法可以采取链地址法。

代码演示:

public class Example {private static final ThreadLocal<User> userThreadLocal = new ThreadLocal<>();public void process() {try {User user = getUserFromRequest();userThreadLocal.set(user);// 业务逻辑} finally {userThreadLocal.remove(); // 确保清理}}
}

        3. 不使用线程池会出现内存泄漏吗?

        如果线程是 普通线程(非线程池中的线程),且任务执行完毕后线程 正常结束(如 run() 方法执行完毕或调用 interrupt()),则:线程对象会被销毁,其内部的 ThreadLocalMap 也会被垃圾回收器回收。无需显式调用 remove(),因为线程结束后,ThreadLocalMap 及其存储的值会自动被释放。

        ThreadLocalMap 是线程的一部分,当线程结束时,整个 ThreadLocalMap 都会被回收,这样即使没有调用 remove(),也不会有泄漏。其次,如果线程没有正确结束,比如长时间运行的守护线程,或者主线程一直运行,这时候可能会有问题。但这种情况比较少见,通常主线程结束程序就会退出了。

        还需要提到静态的 ThreadLocal 实例,如果声明为 static,即使线程结束,ThreadLocal 对象可能不会被回收,但如果是非静态的,随着线程结束,ThreadLocal 对象也会被回收。不过,如果线程没有结束,比如某些情况下线程被长时间持有,比如服务器中的工作线程,这时候即使不使用线程池,也可能导致泄漏。

http://www.xdnf.cn/news/1474615.html

相关文章:

  • 记一次uniapp微信小程序开发scss变量失效的问题
  • 5-10数组元素添加和删除(数组基础操作)
  • 【Python自动化】 21.1 Pandas 读取 Excel 文件的完整指南
  • 从挑西瓜到树回归:用生活智慧理解机器学习算法
  • 【Python】数据可视化之分布图
  • 51单片机---硬件学习(电子琴、主从应答模式、modbus模型、DS18B20传感器显示温度)
  • AI驱动的软件测试:革命性的自动化、缺陷检测与实验优化
  • Java并发机制的底层实现原理
  • 程序化广告快速上手:零基础入门第一课
  • 洛谷 P1591 阶乘数码-普及-
  • PyTorch生成式人工智能——深度分层变分自编码器(NVAE)详解与实现
  • 贪心算法应用:基因编辑靶点选择问题详解
  • 【C++】类和对象(三)
  • Git reset 回退版本
  • stunnel实现TCP双向认证加密
  • Custom SRP - Complex Maps
  • 顺丰,途虎养车,优博讯,得物,作业帮,途游游戏,三七互娱,汤臣倍健,游卡,快手26届秋招内推
  • JVM如何排查OOM
  • 01.单例模式基类模块
  • 微信小程序携带token跳转h5, h5再返回微信小程序
  • Knative Serving:ABP 应用的 scale-to-zero 与并发模型
  • 【Python 】入门:安装教程+入门语法
  • 使用 C# .NETCore 实现MongoDB
  • OpenAI新论文:Why Language Models Hallucinate
  • 【黑客技术零基础入门】2W字零基础小白黑客学习路线,知识体系(附学习路线图)
  • 【C++】C++11的可变参数模板、emplace接口、类的新功能
  • 《云原生微服务治理进阶:隐性风险根除与全链路能力构建》
  • 旧电脑改造服务器1:启动盘制作
  • Element-Plus
  • Nestjs框架: 基于权限的精细化权限控制方案与 CASL 在 Node.js 中的应用实践