Full GC 排查

server/2025/3/1 7:28:07/

在 Java 中,Full GC(完全垃圾回收)会对整个堆(包括年轻代和老年代,甚至可能包括永久代/元空间)进行垃圾回收,通常会导致较长的停顿(STW,Stop-The-World)。如果 Full GC 频繁发生,可能会影响应用的性能,甚至导致 OOM(OutOfMemoryError)。以下是排查 Full GC 的方法和工具:

1. 启用 GC 日志进行分析

(1)JDK 8 及以下版本

在 JVM 启动参数中添加:

-XX:+PrintGCDetails -XX:+PrintGCDateStamps -Xloggc:gc.log

解释:

  • -XX:+PrintGCDetails:输出 GC 详细信息。
  • -XX:+PrintGCDateStamps:打印 GC 发生的时间戳。
  • -Xloggc:gc.log:指定 GC 日志文件。

GC 日志示例:

2025-02-28T10:00:00.123+0000: [Full GC (System.gc())  [PSYoungGen: 1024K->0K(2048K)] [ParOldGen: 4096K->1024K(8192K)], 0.2356780 secs] 

分析重点:

  • Full GC (System.gc()):说明 GC 是由 System.gc() 触发的。
  • ParOldGen: 4096K->1024K(8192K):老年代的使用变化。
  • 0.2356780 secs:GC 停顿时间。

(2)JDK 9 及以上

JDK 9 引入了 统一日志框架(JEP 158),可以使用 -Xlog 进行更细粒度的 GC 日志控制:

-Xlog:gc*:file=gc.log:time,uptime,level,tags

示例:

[2025-02-28T10:00:00.123+0000][info][gc] GC(5) Pause Full (G1 Evacuation Pause) 235ms

2. 使用 jstat 监控 GC

jstat 是 JDK 自带的工具,可以用来实时监控 GC 的情况。

(1)查看 GC 统计信息

jstat -gcutil <pid> 1000

示例输出:

  S0     S1     E      O      M     CCS   YGC    YGCT   FGC   FGCT    GCT0.00  98.65  64.32  92.12  80.45  65.23   150  12.34   20   30.12  42.46

重点关注:

  • O(Old Gen):如果 O 长期接近 100%,说明老年代快满了,可能会触发 Full GC
  • FGC(Full GC 次数):如果这个值增长很快,说明 Full GC 频繁发生。
  • FGCT(Full GC 耗时):观察 Full GC 是否导致了长时间的 STW。

(2)查看具体 GC 详细信息

jstat -gc <pid> 1000

可以看到 Eden、Survivor、Old 区域的内存变化,帮助判断是否因老年代空间不足导致 Full GC

3. 使用 jmap 检查堆内存

如果怀疑 Full GC 频繁发生是由于内存不足,可以用 jmap 生成 堆转储文件(Heap Dump) 进行分析:

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

然后用 Eclipse MATVisualVM 分析内存使用情况,检查是否有内存泄漏或对象无法回收的问题。

此外,可以使用:

jmap -histo <pid>

查看当前堆中的对象分布,重点关注大对象或数量异常的对象。

4. 使用 jconsoleVisualVM 监控

  • JConsole

    jconsole
    
    • 连接到目标 JVM 后,查看 “内存”(Memory) 选项卡,观察老年代的占用情况。
    • 如果 Old Gen 长期接近 100%,可能会触发 Full GC
  • VisualVM

    jvisualvm
    
    • 连接到目标进程,打开 “监视”(Monitor) 选项卡,观察 GC 活动
    • 使用 “Profiler”(分析器) 记录 GC 发生的时间和影响。

5. 检查 Full GC 触发原因

(1)老年代空间不足

  • 观察老年代(Old Gen)占用情况,如果频繁接近 100%,说明 Full GC 可能是由于老年代空间不足导致的。
  • 优化方案
    • 增大老年代:
      -Xmx4g -Xms4g
      
    • 调整 GC 算法(如 G1 或 ZGC):
      -XX:+UseG1GC
      
    • 调整老年代比例:
      -XX:NewRatio=2
      

(2)过多的 System.gc()

  • 某些代码可能显式调用了 System.gc(),强制触发 Full GC
  • 检查代码是否有 System.gc() 调用,可以禁用:
    -XX:+DisableExplicitGC
    

(3)大对象直接进入老年代

  • 如果对象过大,可能会直接分配到老年代,导致 Full GC
  • 优化方案
    • 增大 -XX:PretenureSizeThreshold 以避免大对象直接进入老年代:
      -XX:PretenureSizeThreshold=16m
      
    • 使用 G1GC,可以通过 -XX:InitiatingHeapOccupancyPercent 控制老年代回收时机。

(4)元空间(Metaspace)不足

  • 在 JDK 8+,类元数据存放在 元空间(Metaspace),如果元空间不足也会触发 Full GC
  • 优化方案
    • 增大元空间:
      -XX:MetaspaceSize=512m -XX:MaxMetaspaceSize=1g
      

6. 结论

快速排查 Full GC 的步骤

  1. 启用 GC 日志-Xlog:gc*-XX:+PrintGCDetails)分析 Full GC 触发原因。
  2. 使用 jstat -gcutil <pid> 1000 观察老年代是否满了。
  3. 使用 jmap -histo <pid>jmap -dump 分析堆对象,检查是否有内存泄漏或大对象。
  4. 使用 jconsoleVisualVM 监控 GC 活动,观察 Full GC 频率。
  5. 检查是否有 System.gc() 调用,可以通过 -XX:+DisableExplicitGC 禁用。
  6. 调整 JVM 参数,如 -Xmx, -XX:NewRatio, -XX:MetaspaceSize,或尝试 G1/ZGC。

这样可以快速找出 Full GC 发生的原因,并进行针对性的优化 🚀。


http://www.ppmy.cn/server/171495.html

相关文章

MyBatis基础模块-缓存模块

缓存模块 MyBatis作为一个强大的持久层框架,缓存是其必不可少的功能之一,Mybatis中的缓存分为一级缓存和二级缓存。但本质上是一样的,都是使用Cache接口实现的。缓存位于 org.apache.ibatis.cache包下。 通过结构我们能够发现Cache其实使用到了装饰器模式来实现缓存的处理。…

wordpress子分类调用父分类名称和链接的3种方法

专为导航而生&#xff0c;在wordpress模板制作过程中常常会在做breadcrumbs导航时会用到&#xff0c;子分类调用父分类的名称和链接&#xff0c;下面这段简洁的代码&#xff0c;可以完美解决这个问题。 <?php echo get_category_parents( $cat, true, &raquo; ); ?…

【电路笔记】-MOD计数器

MOD计数器 文章目录 MOD计数器1、概述2、D型触发器3、二分频计数器4、MOD-4 计数器5、模“m”计数器5.1 模5计数器5.2 模 10 计数器6、总结MOD 计数器是级联计数器电路,在复位之前计数到设定的模数值。 1、概述 计数器的工作是通过每个时钟脉冲将计数器的内容前进一个计数来进…

中值滤波结合快速排序算法优化传感器数据预处理

一、算法核心逻辑 目标&#xff1a;在嵌入式系统中&#xff0c;通过快速排序的 “部分排序” 特性&#xff0c;优化中值滤波的计算效率。适用场景&#xff1a;实时传感器数据处理&#xff08;如红外、超声波、加速度计等&#xff09;&#xff0c;窗口大小 N5&#xff08;可根据…

使用逻辑分析仪测量RS485的通讯方法

1. 硬件连接 连接参考地&#xff1a;逻辑分析仪的参考地需要连接到被测设备RS-485收发器的参考地。信号线连接&#xff1a;有以下几种接线方式&#xff1a; 单线连接&#xff1a;将逻辑分析仪的一个信号通道连接到RS-485总线的A端。双线连接&#xff1a;用逻辑分析仪两个信号通…

【六祎 - Note】SQL备忘录;DDL,DML,DQL,DCL

SQL备忘录 from to : 点击访问源地址

9、HTTP/2与HTTP/1.1的区别?【高频】

二进制协议&#xff1a; HTTP/2 不再像 HTTP/1.1 里的纯文本形式的报文&#xff0c;而是全面采用了二进制格式&#xff0c;报文头部和数据体都是二进制&#xff0c;并且统称为帧&#xff08;frame&#xff09;&#xff1a;头信息帧&#xff08;Headers Frame&#xff09;和数据…

使用write函数

使用open命令打开文件后&#xff0c;要往里面写入数据&#xff0c;使用write命令&#xff0c;把buf中count字节的数据写入fd中 关键是&#xff0c;写文件的时候要在这个文件的哪一个位置去写 假如写得时候&#xff0c;文件为空&#xff0c;指针指向最开始的位置&#xff0c;执…