ThreadLocal 会出现内存泄漏吗?

ThreadLocal

ThreadLocal 是一个用来解决线程安全性问题的工具。它相当于让每个线程都开辟一块内存空间,用来存储共享变量的副本。然后每个线程只需要访问和操作自己的共享变量副本即可,从而避免多线程竞争同一个共享资源。它的工作原理很简单(如图)每个线程里面有一个成员变量ThreadLocalMap。当线程访问用 ThreadLocal 修饰的共享数据的时候这个线程就会在自己成员变量 ThreadLocalMap 里面保存一份数据副本。key 指向 ThreadLocal 这个引用,并且是弱引用关系,而 value 保存的是共享数据的副本。因为每个线程都持有一个副本,所以就解决了线程安全性问题。

内存泄漏

ThreadLocal 中的引用关系如图所示(如图),Thread 中的成员变量 ThreadLocalMap,它里面的可以 key 指向 ThreadLocal 这个成员变量,并且它是一个弱引用所谓弱引用,就是说成员变量ThreadLocal 允许在这种引用关系存在的情况下,被 GC回收。一旦被回收,key 的引用就变成了 null,就会导致这个内存永远无法被访问,造成内存泄漏。

总结

不恰当的使用 ThreadLocal,会造成内存泄漏问题。主要原因是,线程的私有变量ThreadLocalMap 里面的 key 是一个弱引用。弱引用的特性,就是不管是否存在直接引用关系, 当成员 ThreadLocal 没用其他的强引用关系的时候,这个对象会被 GC 回收掉。从而导致 key 可能变成 null,造成这块内存永远无法访问,出现内存泄漏的问题。规避内存泄漏的方法有两个:

  • 通过扩大成员变量 ThreadLoca 的作用域,避免被 GC 回收
  • 每次使用完 ThreadLocal 以后,调用 remove 方法移除对应的数据

第一种方法虽然不会造成key为null的现象,但是如果后续线程不再继续访问这个key。也会导致这个内存一直占用不释放,最后造成内存溢出的问题。所以我认为最好是在使用完以后调用 remove 方法移除。

java 复制代码
public class ThreadLocalExample {
    private static ThreadLocal<Integer> myThreadLocal = new ThreadLocal<>();

    public static void main(String[] args) {
        // 创建两个线程,并为每个线程设置不同的值
        Thread thread1 = new Thread(() -> {
            myThreadLocal.set(10);
            System.out.println("Thread 1: " + myThreadLocal.get());
            myThreadLocal.remove(); // 清除线程本地变量的值
        });

        Thread thread2 = new Thread(() -> {
            myThreadLocal.set(20);
            System.out.println("Thread 2: " + myThreadLocal.get());
            myThreadLocal.remove(); // 清除线程本地变量的值
        });

        thread1.start();
        thread2.start();
    }
}
相关推荐
多多*25 分钟前
微服务网关SpringCloudGateway+SaToken鉴权
linux·开发语言·redis·python·sql·log4j·bootstrap
梓仁沐白25 分钟前
【Kotlin】协程
开发语言·python·kotlin
写bug写bug34 分钟前
如何正确地对接口进行防御式编程
java·后端·代码规范
Cyanto1 小时前
Java并发编程面试题
java·开发语言·面试
海的诗篇_1 小时前
前端开发面试题总结-JavaScript篇(一)
开发语言·前端·javascript·学习·面试
在未来等你1 小时前
互联网大厂Java求职面试:AI大模型与云原生技术的深度融合
java·云原生·kubernetes·生成式ai·向量数据库·ai大模型·面试场景
じ☆ve 清风°1 小时前
理解JavaScript中map和parseInt的陷阱:一个常见的面试题解析
开发语言·javascript·ecmascript
sss191s1 小时前
Java 集合面试题从数据结构到 HashMap 源码剖析详解及常见考点梳理
java·开发语言·数据结构
IGP91 小时前
20250606-C#知识:委托和事件
开发语言·c#
LI JS@你猜啊1 小时前
window安装docker
java·spring cloud·eureka