Java导致内存泄漏的多种情况分析

 更新时间:2026年01月09日 08:49:33   作者:烟沙九洲  
本文介绍了Java中常见的内存泄漏情况,包括生命周期长的集合、未关闭的资源连接、ThreadLocal使用不当、内部类与外部类引用非静态内部类、监听器与回调注册后没有注销,推荐使用MAT和VisualVM等工具进行内存泄漏排查,感兴趣的朋友跟随小编一起看看吧

今天我们来一起聊一聊有哪些情况会导致内存泄漏。

什么是 内存泄漏 呢?

内存泄漏 是指对象 已经不再被程序使用,但因为某些原因 无法被垃圾回收器回收,长期占用内存,最终可能引发 OOM(OutOfMemoryError)。

接下来我们看一下常见的几类内存泄漏场景。

1、生命周期长的集合

将对象放入 静态 或 生命周期很长 的集合(如 public static List<Object> list = new ArrayList<>();),即使后面不再需要,集合仍持有其引用,导致无法GC

2、未关闭的资源

连接、流等资源未调用 close() 方法关闭。这些资源不仅占用内存,还可能占用文件句柄(操作系统分配的唯一标识,凭它,你才能操作文件资源)、网络连接等系统资源。比如 数据库连接、文件流(FileInputStream)、Socket连接 等。

public class FileTest {
    public static void main(String[] args) {
        FileInputStream fis = null;
        try {
            fis = new FileInputStream("test.txt");
            // 读取文件,未调用 fis.close()
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } finally {
            // 未调用 fis.close() → fis 持有 Native 引用,无法回收
        }
    }
}

3、ThreadLocal 使用不当

将对象存入 ThreadLocal 后,未在后续调用 remove() 清理。若线程来自线程池(会复用),其 ThreadLocalMap 中的值会一直存活。

public class ThreadLocalTest {
    private static ThreadLocal<User> userThreadLocal = new ThreadLocal<>();
    public static void main(String[] args) {
        // 线程池(核心线程长期存活)
        TThreadPoolExecutor executor = new ThreadPoolExecutor(
                2,
                4,
                10,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<>(100),
                new ThreadFactoryBuilder().setNameFormat("my-thread-pool-%d").setDaemon(false).setPriority(Thread.NORM_PRIORITY).build(),
                new ThreadPoolExecutor.AbortPolicy()
        );
        executor.submit(() -> {
            User user = new User("李四", 30);
            userThreadLocal.set(user); // 存储到 ThreadLocal
            // 业务执行完毕,未调用 remove()
            // 核心线程不会销毁,ThreadLocal 仍持有 user 引用
        });
    }
}

ps:未进行 remove(),还可能会导致 ThreadLocal 取值串门。

4、内部类与外部类引用

非静态内部类(或匿名类)会 隐式持有 外部类的引用。如果内部类实例生命周期更长(如被缓存或另一个线程引用),会阻止外部类被回收。

public class OuterClass {
    private byte[] bigData = new byte[1024 * 1024 * 10]; // 10MB 大对象
    // 非静态内部类
    class InnerClass {
        // 内部类隐式持有 OuterClass 引用
    }
    public InnerClass createInner() {
        return new InnerClass();
    }
    public static void main(String[] args) {
        OuterClass outer = new OuterClass();
        InnerClass inner = outer.createInner();
        // 置空外部类引用,但 inner 仍持有 outer 引用
        outer = null;
        // 若 inner 被静态变量/线程长期持有 → outer 对象(含 bigData)无法回收
    }
}

5、 监听器与回调

注册了 监听器 或 回调 后,在对象不再需要时 没有注销,导致源对象仍持有监听器的引用(比如 事件监听器、消息队列的消费者等)。

排查工具推荐

  • MAT(Memory Analyzer Tool):分析堆 Dump 文件,定位泄漏对象、引用链(谁在持有泄漏对象);
  • VisualVM:JDK 自带工具,监控内存占用趋势,生成堆 Dump,简单排查泄漏。

这里我只列了常见的几种情况,欢迎大家补充其他内存泄漏场景。

到此这篇关于Java导致内存泄漏的多种情况分析的文章就介绍到这了,更多相关java内存泄漏内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 在Intellij Idea中使用jstl标签库的方法

    在Intellij Idea中使用jstl标签库的方法

    这篇文章主要介绍了在Intellij Idea中使用jstl标签库的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-05-05
  • springboot幂等切片的实现

    springboot幂等切片的实现

    本文主要介绍了springboot幂等切片的实现,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-03-03
  • springboot+vue实现阿里云oss上传的示例代码

    springboot+vue实现阿里云oss上传的示例代码

    文件上传是常用的功能,本文主要介绍了springboot+vue实现阿里云oss上传的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-06-06
  • 聊聊Java 成员变量赋值和构造方法谁先执行的问题

    聊聊Java 成员变量赋值和构造方法谁先执行的问题

    这篇文章主要介绍了聊聊Java 成员变量赋值和构造方法谁先执行的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-10-10
  • Java中去除字符串中所有空格的几种方法

    Java中去除字符串中所有空格的几种方法

    这篇文章介绍了Java中去除字符串中所有空格的几种方法,有需要的朋友可以参考一下
    2013-07-07
  • Java类的继承实例详解(动力节点Java学院整理)

    Java类的继承实例详解(动力节点Java学院整理)

    在Java开发中,我们常常用到继承这一概念,可以说继承是Java这类面向对象编程语言的基石,今天小编一起和大家一起学习java类的继承
    2017-04-04
  • Spring Boot 启动端口如何启动

    Spring Boot 启动端口如何启动

    这篇文章主要介绍了Spring Boot 启动端口如何启动的相关资料,需要的朋友可以参考下
    2016-12-12
  • SpringBoot结合ProGuard实现代码混淆(最新版)

    SpringBoot结合ProGuard实现代码混淆(最新版)

    这篇文章主要介绍了SpringBoot结合ProGuard实现代码混淆(最新版),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-10-10
  • Java中的LinkedHashSet解析

    Java中的LinkedHashSet解析

    这篇文章主要介绍了Java中的LinkedHashSet解析,与HashSet不同的是,LinkedHashSet在内部使用了一个双向链表来维护元素的顺序,因此它可以保持元素的插入顺序,这使得LinkedHashSet在需要保持元素顺序的场景下非常有用,需要的朋友可以参考下
    2023-11-11
  • HttpMessageConverter报文信息转换器的深入讲解

    HttpMessageConverter报文信息转换器的深入讲解

    在Spring中内置了大量的HttpMessageConverter,通过请求头信息中的MIME类型,选择相应的HttpMessageConverter,这篇文章主要给大家介绍了关于HttpMessageConverter报文信息转换器的相关资料,需要的朋友可以参考下
    2022-01-01

最新评论