一、背景
ThreadLocal是Java中用于解决多线程共享变量导致的线程安全问题的一种机制。它为每个线程分配一个独立的变量副本,从而避免了线程间的数据竞争。这个我们从上一篇文章《Java面试题:请谈谈对ThreadLocal的理解?》中已经了解。然而,如果使用不当,ThreadLocal也可能导致内存泄露。

那什么是内存泄漏,它和内存溢出有什么区别?

  • 内存溢出(Memory overflow):没有足够的内存提供申请者使用。

  • 内存泄漏(Memory leak):指程序申请内存后,无法释放已申请的内存空间,内存泄漏的堆积终将导致内存溢出。

二、内存泄露案例

以下是一个可能导致ThreadLocal内存泄露的代码示例:

public class ThreadLocalMemoryTest {
// 定义一个静态的ThreadLocal变量
private static final ThreadLocal<LeakyObject> threadLocal = new ThreadLocal<>(); public static void main(String[] args) {
// 创建一个对象并且存储到ThreadLocal中
threadLocal.set(new LeakyObject()); // 强制进行垃圾回收,以便我们可以看到对象是否被回收
System.gc(); try {
// 等待垃圾回收器进行垃圾回收
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
} // 打印MemoryLeakyObject是否被回收的信息
if (LeakyObject.isInstanceActive()) {
System.out.println("LeakyObject instance is still active!");
} else {
System.out.println("LeakyObject instance has been garbage collected.");
} while(true) { }
} // 一个简单的内存泄漏示例类
private static class LeakyObject {
private static boolean instanceActive = true; @Override
protected void finalize() throws Throwable {
super.finalize();
instanceActive = false;
System.out.println("LeakyObject finalize method has been called.");
} public static boolean isInstanceActive() {
return instanceActive;
}
}
}

运行结果如下:

LeakyObject instance is still active!

在这个例子中,我们定义了一个LeakyObject类,它有一个静态变量instanceActive来跟踪对象实例是否仍然存在。重写的finalize方法会在对象被垃圾回收器回收时被调用,并将instanceActive设置为false。

三、代码优化与内存泄露避免

为了解决这个问题,我们需要确保在不再需要ThreadLocal中的数据时释放其占用的内存,从而避免内存泄露。在System.gc();代码执行之前,模拟清除ThreadLocal中的数据。

// 模拟线程退出,应该清除ThreadLocal中的数据
threadLocal.remove();
// 强制进行垃圾回收,以便我们可以看到对象是否被回收
System.gc();

运行结果如下:

LeakyObject finalize method has been called.
LeakyObject instance has been garbage collected.

ThreadLocalMemoryLeakTest类中的main方法模拟了ThreadLocal的使用,并在使用后调用remove方法来清除ThreadLocal中的数据,然后强制进行垃圾回收并等待一段时间,最后检查对象是否被垃圾回收器回收。

四、总结

ThreadLocal作为一种解决多线程共享变量问题的机制,在正确使用的情况下可以提供很高的性能和可靠性。然而,如果不正确使用,它也可能导致内存泄露。
通过了解并避免上述案例中的问题,我们可以更好地利用ThreadLocal来提高应用程序的性能和可靠性。在本次案例中,我们是通过ThreadLocal.remove()方法,来解决内存泄漏问题。

但是其实解决ThreadLocal内存泄漏问题的方法还有2种,需要依据不同的使用场景:

  • 使用不可变对象:ThreadLocal变量存储的对象最好是不可变的,因为不可变的对象不需要频繁更新,也不会因为被多个线程同时修改而出现线程安全问题。如果要修改一个ThreadLocal变量中的对象,最好使用一个新的对象替换原有的对象,从而避免引用泄漏的问题。
  • 使用弱引用:ThreadLocalMap中的弱引用可以保证ThreadLocal实例在当前线程中不再被引用时能够被GC回收,从而防止内存泄漏问题的发生。

Java面试题:细数ThreadLocal大坑,内存泄露本可避免的更多相关文章

  1. ThreadLocal的内存泄露(转)

    ThreadLocal的目的就是为每一个使用ThreadLocal的线程都提供一个值,让该值和使用它的线程绑定,当然每一个线程都可以独立地改变它绑定的值.如果需要隔离多个线程之间的共享冲突,可以使用T ...

  2. ThreadLocal的内存泄露

    ThreadLocal的目的就是为每一个使用ThreadLocal的线程都提供一个值,让该值和使用它的线程绑定,当然每一个线程都可以独立地改变它绑定的值.如果需要隔离多个线程之间的共享冲突,可以使用T ...

  3. Java技术专题之JVM你的内存泄露了吗?

    一.从一个例子开始 关于JVM的内存泄露,让我们从下面一个例子开始吧,大家来判断一下如果按照下面这种分配方式,会不会出现内存泄露呢? import java.util.List; import jav ...

  4. Java有了GC同样会出现内存泄露问题

    1.静态集合类像HashMap.Vector等的使用最容易出现内存泄露,这些静态变量的生命周期和应用程序一致,所有的对象Object也不能被释放,因为他们也将一直被Vector等应用着. Static ...

  5. java程序——CPU过高100%及内存泄露排查

    CPU过高 这类问题可以使用 top 命令观察一些,CPU 是不是都被 Java 程序占用了.比如下面这个截图: 服务器的 CPU 大多都被 Java 占用了.这正是我们之前生产上 CPU 过高的一个 ...

  6. 2019 Java面试题

    马上又是一个金九银十的招聘旺季,小编在这里给大家整理了一套各大互联网公司面试都喜欢问的一些问题或者一些出场率很高的Java面试题,给在校招或者社招路上的你一臂之力. 首先我们需要明白一个事实,招聘的一 ...

  7. java面试题(目录版)

    在https://www.cnblogs.com/marsitman/p/9539369.html  根据自己以往的面试经验,在该基础上做了补充和删减,均链接到相应的地址(链接失效请留言评论). 一. ...

  8. Java 程序的内存泄露问题分析

    什么是内存泄露? 广义的Memory Leak:应用占用了内存,但是不再使用(包括不能使用)该部分内存 狭义的Memory Leak:应用分配了内存,但是不能再获取该部分内存的引用(对于Java,也不 ...

  9. ThreadLocal以及内存泄漏

    ThreadLocal是什么 ThreadLocal 的作用是提供线程内的局部变量,这种变量在线程的生命周期内起作用,减少同一个线程内多个函数或者组件之间一些公共变量的传递的复杂度.但是如果滥用Thr ...

  10. Java读源码之ThreadLocal

    前言 JDK版本: 1.8 之前在看Thread源码时候看到这么一个属性 ThreadLocal.ThreadLocalMap threadLocals = null; ThreadLocal实现的是 ...

随机推荐

  1. matting tool by cs - 视频扣绿幕 1.4 - 软件推荐

    本地软件,还没用,先留个档,试试 第一个下载还套了个广告壳,醉了,但是能用 https://www.123pan.com/s/X3jA-POMQv 这个装了,是原版,没套壳 https://pan.b ...

  2. npm install 的执行顺序,和 安装包的源死磕

    npm install 源的地址加载执行顺序 从近到远 lock文件 这里直接就记录了 包的下载地址 .npmrc 里面的内容 registry=http://registry.npm.xxxx.co ...

  3. 可穿戴设备主控制器芯片AMA3B 源码解析之初体验

    一 背景和缘由   现在的mcu非常多,在超低功耗mcu这块,能赢得市场穿戴式设备使用的产品的确不多,以前是的是stm32 L系列,可是,随着L系列的缺货涨价和技术指标没有快速的发展,很多厂商都抛弃了 ...

  4. UDP、IMCP、ARP协议通过netmap解析的实现。

    上一篇文章我们讲了一个异步的线程池大概需要如何去实现,现在的话,我们如何来解析一个UDP的包. 环境的搭配 这个环境的问题困扰了很久,这个netmap已经不再更新了,支持Ubuntu16.04-Ubu ...

  5. window.showModalDialog与opener及returnValue

    首先来看看 window.showModalDialog 的参数 vReturnValue = window.showModalDialog(sURL [, vArguments] [, sFeatu ...

  6. 大年学习linux(第二节---磁盘管理)

    二.磁盘管理 文件系统配置文件 /etc/filesystems: 系统指定的测试挂载文件系统类型 /proc/filesystems: linux 系统已经加载的文件系统类型 /lib/module ...

  7. 5G+实时云渲染,助力虚拟仿真实训教学升级

    随着新冠疫情走向全球大流行的发展趋势,学校教育被迫迁徙到线上教学平台,供需平衡被打破,疫情让"在线教学"成为"口罩式的刚需". 我们看到互联网+教育带来便利的同 ...

  8. https://codeforces.com/gym/496962

    A略. B最大最小即为答案. C略. DE记录 t 的每个字母 在 s 中出现的最左和最右,特判端点. FG先贪心后反悔 or 背包. *H:不会.AC自动机.

  9. https://codeforces.com/gym/496432

    ABC:略. D 枚举分的段数,然后扫一遍判断. E F G H

  10. Echarts入门案例教程

    一.定义容器变量并获取页面div元素 1 var chartDom = document.getElementById('chart3'); 二.初始化容器 1 var myChart = echar ...