对ThreadLocal内存泄漏及弱引用的理解

 更新时间:2022年1月29日 20:23  点击:373 作者:Burton_J

ThreadLocal内存泄漏及弱引用

1.什么是内存泄漏?Entry的key弱引用与泄漏关系

在TreadLocal中内存泄漏是指TreadLocalMap中的Entry中的key为null,而value不为null。因为key为null导致value一直访问不到,而根据可达性分析,始终有threadRef->currentThread->threadLocalMap->entry->valueRef->valueMemory,导致在垃圾回收的时候进行可达性分析的时候,value可达从而不会被回收掉,但是该value永远不能被访问到,这样就存在了内存泄漏。

因为Entry的key是弱引用,所以在gc的时候key会被回收,而value是强引用,导致value不会被回收。

如果不使用弱引用也会可能会发生内存泄漏,只要在业务代码里,将ThreadLocal的引用置为null,也会导致Entry中value访问不到,但又因为可达,所以gc时候不会被回收,相当于这部分内存资源被浪费了

2.为什么Entry的key使用弱引用

假设threadLocal使用的是强引用,在业务代码中执行threadLocal Instance=null操作,以清理掉threadLocal实例的目的,但是因为threadLocalMap的Entry强引用threadLocal,因此在gc的时候进行可达性分析,threadLocal依然可达,对threadLocal并不会进行垃圾回收,这样就无法真正达到业务逻辑的目的,出现逻辑错误。

假设Entry弱引用threadLocal,尽管会出现内存泄漏的问题,但是在threadLocal的生命周期里(set,getEntry,remove)里,都会针对key为null的脏entry进行处理。

3.预防内存泄漏

ThreadLocal源码中其实已经对内存泄漏问题做了很多优化,在set,get,remove方法中都会对key为null的但是value不为null的Entry进行value置null操作,使得value的引用为null,可达性失败,在gc是可以回收value的内存。

在日常使用中,最后用完TreadLocal后,记得remove,为什么呢?

因为如果不remove,当一次gc执行,这个value就会造成内存泄漏直到当前线程结束(线程结束,ThreaLocalMap会被置为null,而ThreaLocalMap中的Entry自己也就不可达,会被回收,一切都被回收)

线程结束时会执行Thread.exit方法

private void exit() {
    if (group != null) {
        group.threadTerminated(this);
        group = null;
    }
    /* Aggressively null out all reference fields: see bug 4006245 */
    target = null;
    /* Speed the release of some of these resources */
    threadLocals = null;
    inheritableThreadLocals = null;
    inheritedAccessControlContext = null;
    blocker = null;
    uncaughtExceptionHandler = null;
}

匿名内部类会导致内存泄露

内存泄露:就是本该被GC回收的对象,因为各种原因导致的无法被回收,造成内存资源的浪费,从而导致OOM。

如果一个类使用了内部类,而两个类的生命周期不一致,比如内部类的生命周期比外部类生命周期长,

这就会导致外部类的生命周期结束了,本该被回收的,却因为内部类会隐式强引用外部类,所以导致外部类无法被回收,

从而造成了内存泄露。

解决方案

1. 可以避免使用内部类;

2. 内部类可以用弱引用来引用外部类;

3. 使用静态内部类,静态内部类不持有外部类的引用(如果要调用外部类方法或使用外部类属性,可以使用弱引用来解决)。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持猪先飞。

原文出处:https://blog.csdn.net/weixin_37739042/article/details/888688

[!--infotagslink--]

相关文章

  • 详解SpringCloudGateway内存泄漏问题

    这篇文章主要介绍了详解SpringCloudGateway内存泄漏问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-07-16
  • Java基础之内存泄漏与溢出详解

    今天带大家来了解一下Java内存泄漏与溢出的知识,文中有非常详细的介绍,对正在学习Java基础的各位小伙伴呢很有帮助哟,需要的朋友可以参考下...2021-05-15
  • 关于ThreadLocal对request和response的用法说明

    这篇文章主要介绍了关于ThreadLocal对request和response的用法说明,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-08-18
  • Android LeakCanary检测内存泄露原理

    这篇文章主要介绍了分析LeakCanary检测内存泄露原理,帮助大家更好的理解和学习使用Android开发,感兴趣的朋友可以了解下...2021-03-26
  • 详谈C++的内存泄漏问题

    下面小编就为大家带来一篇详谈C++的内存泄漏问题。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...2020-04-25
  • C++程序检测内存泄漏的方法分享

    这篇文章主要介绍了C++程序检测内存泄漏的方法分享,本文讲解了、对象计数、重载new和delete、Hook Windows系统API、使用DiagLeak检测等内容,需要的朋友可以参考下...2020-04-25
  • C#中event内存泄漏总结

    本篇文章给大家分享了在C#中event内存泄漏的相关知识点以及代码分享,有兴趣的朋友参考学习下。...2020-06-25
  • VisualStudio 使用Visual Leak Detector检查内存泄漏

    这篇文章主要介绍了VisualStudio 使用Visual Leak Detector检查内存泄漏的相关资料,需要的朋友可以参考下...2020-04-25
  • Android开发中常见的内存泄漏及避免总结

    Android开发时经常会出现内存泄漏,我想大家应该都遇到过,本文我们总结了Android中常见的内存泄漏原因,以及我们应该如何尽量避免内存泄露。 内存泄漏产生的原因当一...2016-09-20
  • 防IE内存泄漏办法

    <div id="d1"></div> <script > function createButton(){ var obj = document.createElement("button"); obj.innerHTML="点我!"; ...2016-09-20
  • WeakReference(弱引用)让GC需要时回收对象

    我们平常用的都是对象的强引用,如果有强引用存在,GC是不会回收对象的,我们能不能同时保持对对象的引用,而又可以让GC需要的时候回收这个对象呢?本文将为您详细解答,需要了解的朋友可以参考下...2021-09-22
  • Swift源码解析之弱引用

    这篇文章主要给大家介绍了关于Swift源码解析之弱引用的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧...2020-06-30
  • Java中的ThreadLocal功能演示示例

    这篇文章主要介绍了Java中的ThreadLocal功能演示示例,帮助大家更好的理解和使用Java,感兴趣的朋友可以了解下...2021-02-07
  • 关于C语言中弱符号与弱引用的实际应用问题

    在编码过程中,我们经常遇到符号重定义的错误问题,本文通过实例代码展示给大家介绍了C语言弱符号与弱引用的实际应用问题,一起看看吧...2021-09-19
  • C++内存泄漏及检测工具详解

    最简单的方法当然是借助于专业的检测工具,比较有名如BoundsCheck,功能非常强大,相信做C++开发的人都离不开它。此外就是不使用任何工具,而是自己来实现对内存泄露的监控...2020-04-25
  • 详解JAVA 弱引用

    这篇文章主要介绍了 JAVA 弱引用的相关资料,帮助大家更好的理解和学习java引用对象,感兴趣的朋友可以了解下...2020-08-18
  • 避免PHP-FPM内存泄漏导致内存耗尽

    PHP-FPM如果出现存存泄漏问题可能导致服务器内存不足从而导致服务器资源用尽了,这样就会导致网站无法访问了,下面一起来看一篇避免PHP-FPM内存泄漏导致内存耗尽的配置。...2016-11-25
  • Java之ThreadLocal使用常见和方式案例讲解

    这篇文章主要介绍了Java之ThreadLocal使用常见和方式案例讲解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下...2021-08-09
  • 浅谈C语言中的强符号、弱符号、强引用和弱引用

    这篇文章主要介绍了C语言中的强符号、弱符号、强引用和弱引用的定义及相关内容,非常的简单易懂,有需要的朋友可以参考下...2020-04-25
  • 详解Java中的ThreadLocal

    ThreadLocal是JDK包提供的,它提供线程本地变量,如果创建一个ThreadLocal变量,那么访问这个变量的每个线程都会有这个变量的一个副本,在实际多线程操作的时候,操作的是自己本地内存中的变量,从而规避了线程安全问题...2021-06-11