【JUC】二十五、ThreadLocal内存泄漏问题(强软弱虚四种引用)

2023-12-12 20:28

本文主要是介绍【JUC】二十五、ThreadLocal内存泄漏问题(强软弱虚四种引用),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 1、引用之强软弱虚
  • 2、强引用
  • 3、软引用
  • 4、弱引用
  • 5、虚引用
  • 6、ThreadLocal回顾
  • 7、ThreadLocal使用弱引用的原因
  • 8、清除脏Entry
  • 9、最佳实践

不再会被使用的对象或者变量占用的内存不能被回收,就是内存泄露(累积可能导致OOM)。

1、引用之强软弱虚

在这里插入图片描述

  • Reference:强引用
  • SoftReference:软引用
  • WeakReference:弱引用
  • PhantomReference:虚引用

Java 允许使用 finalize方法在垃圾收集器将对象从内存中清除出去之前做必要的清理工作(遗言时机)。

//since Java9,已过期
public class MyObject {@Overrideprotected void finalize() throws Throwable {//finalize用于在对象被不可撤销的丢弃之前执行的操作System.out.println("----invoke finalize method ~");}
}

2、强引用

当内存不足,JVM开始垃圾回收,对于强引用的对象,就算是出现了OOM也不会对该对象进行回收,死都不收。

强引用是最常见的普通对象引用,只要还有强引用指向一个对象,就能表明对象还“活着”,垃圾收集器不会碰这种对象

//eg:
Student student = new Student();
  • 在Java 中最常见的就是强引用,把一个对象赋给一个引用变量,这个引用变量就是一个强引用

  • 当一个对象被强引用变量引用时,它处于可达状态,它是不可能被垃圾回收机制回收的,即使该对象以后永远都不会被用到,JVM也不会回收

  • 因此强引用是造成Java内存泄漏的主要原因之一

对于一个普通的对象,如果没有其他的引用关系,只要超过了引用的作用域或者显式地将相应(强)引用赋值为 null,一般认为就是可以被垃圾收集的了(当然具体回收时机还是要看垃圾收集策略)。

public class ReferenceDemo {public static void main(String[] args) {MyObject myObject = new MyObject();System.out.println("gc before: " + myObject);myObject = null;//手动触发一次GCSystem.gc();System.out.println("gc after: " + myObject);}
}

在这里插入图片描述

调用finalize方法是另一线程,这里的打印顺序不用关注。

3、软引用

软引用是一种相对强引用弱化了一些的引用,需要用java.lang.ref.SoftReference类来实现,可以让对象豁免一些垃圾收集。

总之就是相对强引用而言,稍微松一点,GC触发时:

  • 当系统内存充足时,它不会被回收
  • 当系统内存不足时,它会被回收

软引用通常用在对内存敏感的程序中,比如高速缓存就有用到软引用,内存够用的时候就保留,不够用就回收。用SoftReference把自定义对象包装一下,对应的引用就变成了软引用。

SoftReference<MyObject> softReference = new SoftReference<>(new MyObject());
System.out.println("gc before: " + softReference.get());
//手动触发一次GC
System.gc();
System.out.println("gc after: " + softReference.get());

在这里插入图片描述

修改Demo类的JVM内存限制,创造一个内存不足的情况:

在这里插入图片描述

创建一个20M的数组,超过了上面的最大内存,模拟内存不足,对象被回收:
在这里插入图片描述

在这里插入图片描述

4、弱引用

对于只有弱引用的对象来说,只要垃圾回收机制一运行,不管JVM的内存空间是否足够,都会回收该对象占用的内存。

WeakReference<MyObject> weakReference = new WeakReference<>(new MyObject());
System.out.println("gc before: " + weakReference.get());
System.gc();
System.out.println("gc after: " + weakReference.get());

在这里插入图片描述

软引用和弱引用的适用场景举例:

假如有一个应用需要读取大量的本地图片,如果每次读取图片都从硬盘读取则会严重影响性能,如果一次性全部加载到内存中又可能造成内存溢出,此时使用软引用可以解决这个问题。

设计思路是:

用一个HashMap来保存图片的路径和相应图片对象关联的软引用之间的映射关系,在内存不足时,JVM会自动回收这些缓存图片对象所占用的空间,从而有效地避免了OOM的问题。

Map<String, SoftReference<Bitmap>> imageCache = new HashMap<String, SoftReference<Bitmap>>().

5、虚引用

1)虚引用必须和引用队列ReferenceQueue联合使用

  • 虚引用需要java.lang.ref.PhantomReference类来实现
  • 虚,即形同虚设
  • 虚引用不会决定对象的生命周期
  • 如果一个对象仅持有虚引用,则它和没任何引用一样,随时都可能被垃圾回收器回收
  • 不能单独使用,也不能通过它访问对象
  • 虚引用必须和引用队列ReferenceQueue联合使用,如果虚引用对象被干掉了,就装到队列里

2)PhantomReference虚引用的get方法总是返回null

  • 虚引用的主要作用是跟踪对象被垃圾回收的状态,仅仅是提供了一种确保对象被 finalize以后,做某些事情的通知机制
  • PhantomReference的get方法总是返回null,因此无法访问对应的引用对象

3)处理监控通知使用

  • 设置虚引用关联对象的唯一目的,就是在这个对象被收集器回收的时候收到一个系统通知或者后续添加进一步的处理,用来实现比finalize机制更灵活的回收操作

构造方法:

//传入要包装的对象和引用队列
PhantomReference(T referent, ReferenCeQueue<? super T> queue)

继续设置JVM最大内存10M:

public class ReferenceDemo {public static void main(String[] args) {MyObject myObject = new MyObject();ReferenceQueue<MyObject> referenceQueue = new ReferenceQueue<>();PhantomReference<MyObject> phantomReference = new PhantomReference<>(myObject, referenceQueue);List<byte[]> list = new ArrayList<>();new Thread(() -> {while (true){list.add(new byte[1024 * 1024]); //1M//歇500ms,写1M进Listtry {TimeUnit.MILLISECONDS.sleep(500);} catch (InterruptedException e) {e.printStackTrace();}//验证下每次get都是nullSystem.out.println(phantomReference.get() + " list add OK.");}},"t1").start();new Thread(() -> {while (true){Reference<? extends MyObject> reference = referenceQueue.poll();if(reference != null){System.out.println("有虚引用对象被回收,加入了队列");//break;}}},"t2").start();}}

开一个线程去占用内存,另开一个线程去查看队列,可以看到中途虚引用对应的对象被回收时,会加入到队列中。

在这里插入图片描述

在这里插入图片描述

6、ThreadLocal回顾

在这里插入图片描述

ThreadLocal是一个壳子,真正的存储结构是ThreadLocal里的ThreadLocalMap这个内部类,每个Thread对象维护着ThreadLocalMap的引用,ThreadLocalMap则用Entry来进行存储。

  • 调用ThreadLocal的set方法时,实际上就是往ThreadLocalMap设置值,key是ThreadLoca对象,值Value是传递进来的对象
  • 调用ThreadLocal的get方法时,实际上就是往ThreadLocalMap获取值,key是ThreadLocal对象

ThreadLocal本身并不存储值(ThreadLocal是一个壳),它只是自己作为一个key来让线程从ThreadLocalMap获取value。正因为这个原理,所以ThreadLocal能够实现线程间的"数据隔离",获取当前线程的局部变量值,不受其他线程影响

7、ThreadLocal使用弱引用的原因

public void function01(){//新建一个ThreadLocal对象,t1是强引用指向这个对象ThreadLocal<String> t1 = new ThreadLocal<>();//实际是创建了一个Entry对象,根据Entry源码知:Entry对象里的key(即ThreadLocal)是弱引用指向这个对象//当一个ThreadLocal实例对象只被Entry类实例(或者其它弱引用实例)引用时,它就会被GC回收t1.set("code9527");t1.get();
}

在这里插入图片描述

当function1方法执行完毕后,栈帧销毁,强引用 t1 也就没有了。但此时线程的ThreadLocalMap里某个entry的key引用还指向这个对象。此时:

  • 若这个key引用是强引用,就会导致key指向的ThreadLocal对象,以及v指向的对象不能被gc回收,造成内存泄漏

  • 若这个key引用是弱引用就大概率会减少内存泄漏的问题(还有一个key为nul的雷,在下面再展开)。

使用弱引用,就可以使ThreadLocal对象在方法执行完毕后顺利被回收且Entry的key引用指向为null,而此后我们调用get、set、remove方法时,就会尝试删除key为null的entry,可以释放value对象所占用的内存。

8、清除脏Entry

当我们为threadLocal变量赋值,实际上就是当前Entry(threadLocal实例为key,值为value)往这个threadLocalMap中存放。Entry中的key是弱引用,当threadLocal外部强引用被置为null(比如前面例子的t1=null),那么系统 GC 的时候,根据可达性分析,这个threadLocal实例就没有任何一条路能够引用到它, 这个ThreadLocal势必会被回收。

这样一来,ThreadLocalMap中就会出现key为nul的Entry,就没有办法访问这些key为nul的Entry的value,如果当前线程再迟迟不结束的话(线程池,线程在不断复用),这些key为null的Entry的value就会一直存在一条强引用:某个线程池中线程T1的引用Thread Ref ⇒ Thread ⇒ ThreadLocalMap ⇒ Entry ⇒ value ,因此永远无法回收,最后造成内存泄漏。

当然,如果当前thread运行结束,threadLocal、threadLocalMap、Entry没有引用链可达,在垃圾回收的时候都会被系统进行回收。

关于以上key为null的脏Entry的清除 ===> expungeStaleEntry方法

其中,get、set、remove等方法源码中,都有调用expungeStaleEntry方法,如get --> 调getEntry方法 --> getEntryAfterMiss方法:

在这里插入图片描述

虽然弱引用,保证了key指向的ThredLocal对象能被及时回收,但是v指向的value对象是需要ThreadLocalMap调用get、set时发现key为nul时才会去回收整个entry、value,因此弱引用不能100%保证内存不泄露,我们要在不使用某个ThreadLocal对象后,手动调用remoev方法来删除它,尤其是在线程池中,不仅仅是内存泄露的问题,因为线程池中的线程是重复使用的,意味着这个线程的ThreadLocalMap对象也是重复使用的,如果我们不手动调用remove方法,那么后面的线程就有可能获取到上个线程遗留下来的value值,造成bug。

总结:

  • 弱引用保证ThreadLocal对象被及时回收,key为null的Entry会累积
  • get、set时检查所有键为null的Entry对象并删除

9、最佳实践

  • 【建议】创建ThreadLocal对象采用静态方法ThreadLocal.withInitial(() -> 初始值)
  • 【建议】把ThreadLocal修饰为static(若某个属性所有对象都相同,则用静态变量,存方法区,如国籍,这样只在方法区保存一份,可避免不必要的内存空间浪费,反之,则是实例变量)
  • 【强制】用完手动remove

在这里插入图片描述

最后,对ThreadLocal的总结:

  • ThreadLocal 并不解决线程间共享数据的问题
  • ThreadLocal 适用于变量在线程间隔离且在方法间共享的场景
  • ThreadLocal 通过隐式的在不同线程内创建独立实例副本避免了实例线程安全的问题
  • 每个线程持有一个只属于自己的专属Map并维护了ThreadLocal对象与具体实例的映射,该Map由于只被持有它的线程访问,故不存在线程安全以及锁的问题
  • ThreadLocalMap的Entry对ThreadLocal的引用为弱引用,避免了ThreadLocal对象无法被回收的问题
  • 都会通过expungeStaleEntry、cleanSomeSlots、replaceStaleEntry这三个方法回收键为 null 的 Entry对象的值 以及 Entry 对象本身,从而防止内存泄漏,属于安全加固的方法
  • 群雄逐鹿起纷争,人各一份天下安

这篇关于【JUC】二十五、ThreadLocal内存泄漏问题(强软弱虚四种引用)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/485820

相关文章

C#中读取XML文件的四种常用方法

《C#中读取XML文件的四种常用方法》Xml是Internet环境中跨平台的,依赖于内容的技术,是当前处理结构化文档信息的有力工具,下面我们就来看看C#中读取XML文件的方法都有哪些吧... 目录XML简介格式C#读取XML文件方法使用XmlDocument使用XmlTextReader/XmlTextWr

mybatis和mybatis-plus设置值为null不起作用问题及解决

《mybatis和mybatis-plus设置值为null不起作用问题及解决》Mybatis-Plus的FieldStrategy主要用于控制新增、更新和查询时对空值的处理策略,通过配置不同的策略类型... 目录MyBATis-plusFieldStrategy作用FieldStrategy类型每种策略的作

linux下多个硬盘划分到同一挂载点问题

《linux下多个硬盘划分到同一挂载点问题》在Linux系统中,将多个硬盘划分到同一挂载点需要通过逻辑卷管理(LVM)来实现,首先,需要将物理存储设备(如硬盘分区)创建为物理卷,然后,将这些物理卷组成... 目录linux下多个硬盘划分到同一挂载点需要明确的几个概念硬盘插上默认的是非lvm总结Linux下多

Python Jupyter Notebook导包报错问题及解决

《PythonJupyterNotebook导包报错问题及解决》在conda环境中安装包后,JupyterNotebook导入时出现ImportError,可能是由于包版本不对应或版本太高,解决方... 目录问题解决方法重新安装Jupyter NoteBook 更改Kernel总结问题在conda上安装了

golang内存对齐的项目实践

《golang内存对齐的项目实践》本文主要介绍了golang内存对齐的项目实践,内存对齐不仅有助于提高内存访问效率,还确保了与硬件接口的兼容性,是Go语言编程中不可忽视的重要优化手段,下面就来介绍一下... 目录一、结构体中的字段顺序与内存对齐二、内存对齐的原理与规则三、调整结构体字段顺序优化内存对齐四、内

pip install jupyterlab失败的原因问题及探索

《pipinstalljupyterlab失败的原因问题及探索》在学习Yolo模型时,尝试安装JupyterLab但遇到错误,错误提示缺少Rust和Cargo编译环境,因为pywinpty包需要它... 目录背景问题解决方案总结背景最近在学习Yolo模型,然后其中要下载jupyter(有点LSVmu像一个

解决jupyterLab打开后出现Config option `template_path`not recognized by `ExporterCollapsibleHeadings`问题

《解决jupyterLab打开后出现Configoption`template_path`notrecognizedby`ExporterCollapsibleHeadings`问题》在Ju... 目录jupyterLab打开后出现“templandroidate_path”相关问题这是 tensorflo

SpringBoot中使用 ThreadLocal 进行多线程上下文管理及注意事项小结

《SpringBoot中使用ThreadLocal进行多线程上下文管理及注意事项小结》本文详细介绍了ThreadLocal的原理、使用场景和示例代码,并在SpringBoot中使用ThreadLo... 目录前言技术积累1.什么是 ThreadLocal2. ThreadLocal 的原理2.1 线程隔离2

如何解决Pycharm编辑内容时有光标的问题

《如何解决Pycharm编辑内容时有光标的问题》文章介绍了如何在PyCharm中配置VimEmulator插件,包括检查插件是否已安装、下载插件以及安装IdeaVim插件的步骤... 目录Pycharm编辑内容时有光标1.如果Vim Emulator前面有对勾2.www.chinasem.cn如果tools工

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动