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();
    }
}
相关推荐
Michaelwubo8 小时前
swagger全集通+mock(prism)
java
叶子野格9 小时前
《C语言学习:文件操作》16
c语言·开发语言·c++·学习·visual studio
AI科技星9 小时前
万有引力G与真空介电常数ε0全维度完整关系式汇编(基于v=c螺旋时空理论)
c语言·开发语言·前端·javascript·网络·汇编·electron
ss2739 小时前
ai编程Trae cn生成图书管理系统(1)
java·数据库·spring boot·python·flask·fastapi
如竟没有火炬9 小时前
寻找峰值——二分
java·开发语言·数据结构·python·算法·散列表
Lumbrologist9 小时前
【C++】零基础入门 · 第 17 节:多线程编程基础
java·c++·算法
tkevinjd9 小时前
事务、ACID与隔离
java·数据库·sql
阿洛学长9 小时前
Kali Linux 虚拟机安装(VMware Workstation 17)
java·linux·服务器
AI人工智能+电脑小能手9 小时前
【大白话说Java面试题 第91题】【Mysql篇】第21题:分布式锁的使用场景和原理?
java·数据库·分布式·mysql·面试
JAVA社区9 小时前
Java高级全套教程(十三)—— 分布式锁超详细实战详解(原理+三种方案企业级落地)
java·开发语言·分布式·spring cloud·面试·java-zookeeper