ThreadLocal原理及其内存泄漏

ThreadLocal通过为每个线程创建一个共享变量的副本来保证各个线程之间变量的访问和修改互不影响。

ThreadLocal存放的值是线程内共享的,线程间互斥的,主要用于线程内共享数据,避免通过参数传递。

ThreadLocal有四个方法:

initialValue:返回此线程局部变量的初始值。

get:返回此线程局部变量的当前线程副本的值。如果线程第一次调用该方法,则创建并初始化此副本。

set:将当前线程副本的值设置为指定值。

remove:移除此线程局部变量的值。

Thread内部有两个变量,threadLocals和inheritableThreadLocals

ThreadLocal内部有个静态内部类ThreadLocalMap,而在外部没办法获取到这个类,每次操纵都需要通过ThreadLocal。而这个map里面维护了一个数据结构为Entry的数组,key就是ThreadLocal,value就是ThreadLocal的value;,而且这个引用还是个弱引用;具体节点类型如下

所以当系统中存在多个线程的时候,具体的引用是这个样子的(手头画图工具太难用,手快一些,将就看吧):

首先最上面一条引用线路,threadLocal对象,这是个弱引用如果没有其他引用这个就会被回收。

但是,下面两条引用都是强引用,进行GC的时候进行标记,只要线程不消亡threadlocal还是可达的;

如果是不用线程池的话,这里随着线程的消亡,下面两个引用也就没了,GC就会回收掉该片区域,但是现住基本上线程都是在线程池里的,如果一个线程会一直活跃不会消亡,在这种情况下,如果在生命周期结束的时候没有去显式的的清理变量(在线程任务完成时调用ThreadLocal的remove方方法),那线程内部的引用会依然存在,虽然这时候threadlocal已经是null了,但是对应的value依然存在,这时候线程内部里的threadlocalmap里就会存在一个key为null的Entry,因为key是null,所有这块数据永远都不会被访问的,这就是内存泄漏了。

相关推荐
超梦dasgg12 分钟前
Java 生产环境 MQ 技术选型全解析
java·开发语言·java-rocketmq·java-rabbitmq
霸道流氓气质12 分钟前
Spring AI 多工具链式调用(Tool Chain)极简实战
java·人工智能·spring
罗超驿1 小时前
22.深入剖析JDBC架构:从原生API到企业级数据交互核心
java·数据库·mysql·面试
一直有一个ac的梦想1 小时前
cmu15445 2025fall lec 18 transactions with two-phase lock
java·开发语言·数据库
九皇叔叔1 小时前
Spring-Ai-Alibaba [04] 04-llm-platform-custom-demo
java·人工智能·spring
技术路上的探险家1 小时前
Sa-Token 单点登录(SSO)三种模式大白话详解:告别重复登录
java·sa-token·单点登录·sso
JAVA社区1 小时前
Java进阶全套教程(四)—— SpringMVC框架详解
java·开发语言·spring·面试·职场和发展
ㄣ知冷煖★1 小时前
统一网关架构实践:从 Token 鉴权到路由、策略与凭证池转发全链路解析
java·服务器·架构
Lumbrologist1 小时前
【C++】零基础入门 · 第 2 节:变量、基本数据类型与输入输出
java·开发语言·c++
GISer_Jing1 小时前
Three.JS渲染架构解读
java·javascript·架构