Java内存泄漏早期排查流程(未触发频繁Full GC前)

 更新时间:2026年06月05日 09:39:11   作者:码不停蹄的玄黓  
内存泄漏早期最典型特征:老年代/堆内存呈缓慢、持续、不可逆的上涨,但还没严重到频繁Full GC、OOM,本文给大家介绍了一套生产环境可直接落地的排查流程,从简单到深入,不影响业务,需要的朋友可以参考下

内存泄漏早期最典型特征老年代/堆内存呈缓慢、持续、不可逆的上涨,但还没严重到频繁Full GC、OOM。
这个阶段最容易定位根因,核心思路是:监控趋势 → 定位泄漏区域 → 抓取堆快照 → 分析泄漏对象 → 定位代码

下面给你一套生产环境可直接落地的排查流程,从简单到深入,不影响业务。

一、第一步:先确认「真的是内存泄漏」

先排除正常内存使用、堆设置过小等情况。

1. 看内存增长曲线(最关键)

通过监控看 堆内存/老年代 趋势:

  • 正常:GC 后内存会大幅回落,曲线呈锯齿状
  • 泄漏:每次GC后回落很少,整体持续缓慢上涨

2. 快速命令验证(无需工具)

jstat -gcutil <pid> 1000 10  # 每1秒输出一次GC情况,共10次

重点看 3 个指标:

  • O:老年代使用率
  • FGC:Full GC 次数
  • FGCT:Full GC 总耗时

泄漏早期特征

  • O 持续缓慢上升(比如从 30% → 40% → 50%)
  • FGC 很少(0~几次)
  • 每次Full GC后,老年代下降幅度极小

只要满足「老年代只涨不跌」,基本就是内存泄漏

二、第二步:定位泄漏类型(区分堆内/堆外)

绝大多数泄漏是 堆内泄漏,极少数是堆外。

1. 堆内泄漏(95%场景)

  • 特征:老年代持续上涨
  • 原因:对象被长生命周期引用持有(静态集合、线程池、缓存、单例)

2. 堆外泄漏

  • 特征:堆内存正常,但进程物理内存持续涨
  • 原因:NIO DirectByteBuffer、JNI、池化资源未释放

先按堆内泄漏排查,效率最高。

三、第三步:无侵入抓取堆快照(核心步骤)

不需要重启、不需要压测、低影响,生产直接用

推荐抓取时机

  • 一次Full GC之后立即抓(最干净,能精准定位泄漏对象)
  • 或者内存涨到相对高位时抓

抓取命令

jmap -dump:format=b,file=leak.hprof <pid>
  • 生成 leak.hprof 二进制堆转储文件
  • 大小≈当前堆使用量,注意磁盘空间

小提示:JDK8+ 加 -dump:live 可以只抓存活对象,文件更小:

jmap -dump:live,format=b,file=leak.hprof <pid>

四、第四步:用工具分析泄漏(10分钟定位代码)

最常用、最稳定的工具:Eclipse MAT(免费、强大)

1. MAT 打开 hprof 文件

选择 Acquire Heap Dump / Open Heap Dump

2. 直接使用「泄漏嫌疑报告」

MAT 会自动生成:Leak Suspects Report
它会直接告诉你:

  • 哪个类占用内存最大
  • 被谁引用
  • 哪行代码导致泄漏

3. 核心分析 3 步

  1. Histogram(直方图)
    看对象实例数,找数量异常多的业务对象(你的自定义类)
  2. Merge Shortest Paths to GC Roots
    右键对象 → 查看 GC Root 引用链
  3. 排除弱引用/软引用
    只看 强引用链,泄漏一定在这里

泄漏对象的典型特征

  • 你的业务实体类(User/Order/Config等)实例数只增不减
  • static Map/List/Set 持有
  • 线程池 / 单例 / 缓存 持有
  • ThreadLocal 持有(高频泄漏点)

五、第五步:早期泄漏最常见的 5 种代码根因

你在分析时,优先排查这 5 类代码,90% 泄漏都在这里:

1. 静态集合(最常见)

private static List<Object> list = new ArrayList<>();
// 只add不remove,对象永远无法回收

2. ThreadLocal 未清理

private static ThreadLocal<Object> tl = new ThreadLocal<>();
// 线程池中的线程长期存活,ThreadLocalMap 一直持有对象

3. 线程池/自定义线程忘记关闭

  • 线程是 GC Root,线程里的引用全部泄漏
  • 线程池任务里持有大对象,且任务无限等待

4. 缓存未设置过期/淘汰

private static Map<String, Object> CACHE = new HashMap<>();
// 无过期、无淘汰,无限增长

5. IO/连接未关闭

  • InputStream、Socket、Connection 未 close
  • 底层持有堆外/堆内引用

六、无工具、命令行快速初筛(应急用)

如果不能用MAT,可以先用命令粗定位:

jmap -histo <pid> | head -20

输出:实例数、字节数、类名
自定义类、实例数异常高的类,就是怀疑对象。

七、最佳实践:提前发现早期泄漏

不用等问题爆发,配置监控即可:

  1. 监控 老年代使用率趋势
  2. 监控 GC后内存回落幅度
  3. 配置告警:老年代连续 10 次 GC 不下降
  1. 看趋势:老年代只涨不跌 = 泄漏
  2. 抓快照:Full GC 后 jmap 抓 hprof
  3. 用MAT:看 Leak Suspects
  4. 查引用:静态集合、ThreadLocal、线程池、缓存
  5. 看代码:谁长期持有对象不释放

以上就是Java内存泄漏早期排查流程(未触发频繁Full GC前)的详细内容,更多关于Java内存泄漏早期排查的资料请关注脚本之家其它相关文章!

相关文章

  • @validated注解异常返回JSON值方式

    @validated注解异常返回JSON值方式

    这篇文章主要介绍了@validated注解异常返回JSON值方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-10-10
  • SpringBoot + 微信公众号JSAPI支付功能的实现

    SpringBoot + 微信公众号JSAPI支付功能的实现

    这篇文章主要介绍了SpringBoot + 微信公众号JSAPI支付功能的实现,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-03-03
  • Mybatis分步查询的实现示例

    Mybatis分步查询的实现示例

    本文主要介绍了Mybatis分步查询的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-07-07
  • Java的MyBatis框架中对数据库进行动态SQL查询的教程

    Java的MyBatis框架中对数据库进行动态SQL查询的教程

    这篇文章主要介绍了Java的MyBatis框架中对数据库进行动态SQL查询的教程,讲解了MyBatis中一些控制查询流程的常用语句,需要的朋友可以参考下
    2016-04-04
  • SpringBoot中使用Redis Stream实现消息监听示例

    SpringBoot中使用Redis Stream实现消息监听示例

    本文主要介绍了SpringBoot中使用Redis Stream实现消息监听示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-06-06
  • idea如何生成springboot单元测试用例

    idea如何生成springboot单元测试用例

    这篇文章主要介绍了idea生成springboot单元测试用例,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-08-08
  • Spring Boot项目使用WebClient调用第三方接口的详细教程(附实例代码)

    Spring Boot项目使用WebClient调用第三方接口的详细教程(附实例代码)

    WebClient是Spring WebFlux模块提供的一个非阻塞的基于响应式编程的进行Http请求的客户端工具,从Spring5.0开始提供,这篇文章主要介绍了Spring Boot项目使用WebClient调用第三方接口的相关资料,需要的朋友可以参考下
    2025-09-09
  • 简述mybatis大于小于的转义

    简述mybatis大于小于的转义

    这篇文章主要介绍了mybatis大于小于的转义以及xml中常用转义字符,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2016-09-09
  • java中的hashCode方法小例子

    java中的hashCode方法小例子

    这篇文章主要介绍了java中的hashCode方法小例子,有需要的朋友可以参考一下
    2013-12-12
  • Springboot 配置RabbitMQ文档的方法步骤

    Springboot 配置RabbitMQ文档的方法步骤

    这篇文章主要介绍了Springboot 配置RabbitMQ文档的方法步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-03-03

最新评论