当前位置: 首页 > news >正文

GC 频率和触发条件

在 Java 中,垃圾回收(GC)的频率触发条件取决于 GC算法、堆内存分配、对象生命周期 以及 JVM参数 的配置。下面详细介绍这些影响因素:

1. GC 触发条件

GC 主要触发的情况如下:

(1) 年轻代 GC(Minor GC / Young GC)

触发条件

  • Eden 区满了:当新对象分配到 Eden 区,如果 Eden 区没有足够的空间分配新对象,就会触发 Minor GC。
  • Survivor 空间不足:当存活对象从 Eden 复制到 Survivor,但 Survivor 空间不够时,也可能导致 Minor GC。

特点

  • 仅回收 年轻代(Young Generation),不会影响老年代(Old Generation)。
  • 采用复制算法(如 Serial、Parallel、G1 的 YGC)。
  • 停顿时间短,但回收频率较高。
  • Minor GC 之后,存活对象可能晋升到老年代

(2) 老年代 GC(Major GC / Old GC)

触发条件

  • 老年代空间不足:当对象从 Survivor 晋升到老年代,或者大对象直接进入老年代,导致老年代空间不够时,会触发 Major GC。
  • CMS GC 的 concurrent mode failure:CMS GC 在并发回收过程中如果老年代空间不足,会触发 STW 的 Full GC。
  • G1 GC 触发 Mixed GC:G1 在一定条件下会触发回收老年代的 Mixed GC。

特点

  • 主要清理 老年代(Old Generation),回收存活时间较长的对象。
  • 相比 Minor GC,Major GC 的停顿时间更长,但一般回收频率较低。
  • 某些 GC(如 CMS)不会 STW,而是并发执行(Concurrent Mark-Sweep)。

(3) Full GC

触发条件

  • 显式调用 System.gc()(不推荐,因为 JVM 可能会忽略)。
  • 老年代空间不足:当老年代没有足够空间存放新对象时,Major GC 可能变成 Full GC。
  • Metaspace/元空间溢出(如类加载过多,导致 java.lang.OutOfMemoryError: Metaspace)。
  • CMS GC 失败:如果 CMS GC 过程中发生 concurrent mode failure,会触发 Full GC。
  • G1 GC 触发 Full GC:当 G1 发现回收无法跟上对象分配速度时,会进行 STW 的 Full GC。

特点

  • 回收整个堆(包括年轻代 + 老年代 + 元空间)
  • 停顿时间长,影响系统吞吐量和响应时间。
  • 一般不希望频繁发生 Full GC,需要调优。

2. GC 频率的影响因素

GC 的触发频率取决于以下几个因素:

(1) 对象分配速率

  • 短生命周期对象多(临时变量、业务请求数据)Minor GC 频繁
  • 大量大对象(如 byte[] → 可能直接进入老年代,加速 Major/Full GC

(2) GC 算法

不同 GC 算法对 GC 频率的影响不同:

  • Serial GC(单线程、适用于小内存) → GC 频率高,暂停时间长。
  • Parallel GC(多线程 GC,吞吐量优先) → GC 频率较低,适用于高吞吐场景。
  • G1 GC(区域化分代、回收预测) → 控制 GC 停顿时间,适用于大内存。
  • ZGC、Shenandoah GC(低延迟 GC) → 减少 GC 影响,适用于大内存应用。

(3) JVM 参数

JVM 相关参数直接影响 GC 频率:

  • -Xms / -Xmx(堆内存大小):
    • 较小的堆内存 → GC 触发更频繁。
    • 较大的堆内存 → GC 触发较少,但可能增加 Full GC 停顿时间。
  • -XX:NewRatio(年轻代与老年代的比例):
    • 较大年轻代 → Minor GC 频率降低,但可能加速老年代填满导致 Major GC 。
    • 较小年轻代 → Minor GC 频率上升,但老年代增长较慢。
  • -XX:SurvivorRatio(Eden 和 Survivor 的比例):
    • Survivor 较小 → 对象更容易晋升老年代,加快 Major GC 触发。
    • Survivor 较大 → Minor GC 次数可能减少,但 Survivor 可能浪费空间。
  • -XX:MaxTenuringThreshold(晋升老年代的阈值):
    • 较低阈值 → 对象更快晋升老年代,可能增加 Major GC 频率。
    • 较高阈值 → 对象更长时间停留在 Survivor,可能增加 Minor GC 频率。

(4) GC 负担

  • 对象回收速率低 → GC 触发频率更高。
  • 对象生命周期较长(长生命周期的缓存对象等) → 老年代更容易被填满,增加 Major/Full GC 频率。

3. 如何优化 GC 频率

(1) 调整堆内存大小

  • 增大 -Xmx(最大堆内存),减少 GC 触发频率。
  • 增大 -Xms(初始堆内存),减少动态扩展导致的 Full GC。

(2) 调整 GC 参数

  • 增加年轻代大小-XX:NewRatio=1):减少 Minor GC 触发频率,但可能影响老年代回收。
  • 调整 Survivor 空间-XX:SurvivorRatio=6):减少对象晋升到老年代,降低 Major GC 频率。
  • 调高 -XX:MaxTenuringThreshold(如 10),避免短生命周期对象过早进入老年代。

(3) 选择合适的 GC 算法

  • 吞吐量优先(如并发任务多、批量计算) → Parallel GC-XX:+UseParallelGC)。
  • 低延迟场景(如微服务、高并发请求) → G1 GC-XX:+UseG1GC)。
  • 极低延迟需求(如金融系统)ZGC/Shenandoah GC-XX:+UseZGC-XX:+UseShenandoahGC)。

(4) 监控 GC

  • 开启 GC 日志-Xlog:gc*-XX:+PrintGCDetails)观察 GC 频率。
  • 使用 jstat 分析 GC
    jstat -gcutil <pid> 1000
    
  • 使用 VisualVM、Arthas 监控 GC 状态

4. 总结

GC 类型触发条件影响
Minor GC (Young GC)Eden 区满,Survivor 区空间不足频率高,暂停时间短,对业务影响小
Major GC (Old GC)老年代空间不足频率较低,暂停时间长,对吞吐量影响较大
Full GC老年代不足、Metaspace 溢出、CMS 失败等影响最大,应尽量避免

优化 GC 频率的核心 是合理分配堆内存、调整 GC 策略,并监控 GC 运行情况。


文章转载自:

http://nTd1G0R0.bxqry.cn
http://1x9TDsVZ.bxqry.cn
http://Hop3wU0k.bxqry.cn
http://RdE3x8d4.bxqry.cn
http://AbUGZJdg.bxqry.cn
http://iqiQsChJ.bxqry.cn
http://z26JfNOI.bxqry.cn
http://Plq020j7.bxqry.cn
http://WOSpborU.bxqry.cn
http://XTe45Mb8.bxqry.cn
http://ikBLDrJk.bxqry.cn
http://cIQ5QFaK.bxqry.cn
http://hCmgIhup.bxqry.cn
http://Ig2IT77c.bxqry.cn
http://Rh3fwUl4.bxqry.cn
http://3bU1f5ij.bxqry.cn
http://qdSQG4Hx.bxqry.cn
http://cmz4WVBV.bxqry.cn
http://RSrmiHq9.bxqry.cn
http://yCzgMZFU.bxqry.cn
http://LYa43tvV.bxqry.cn
http://66x5dPJ1.bxqry.cn
http://qUVwZclE.bxqry.cn
http://wJzhs3I6.bxqry.cn
http://UvJFHXsd.bxqry.cn
http://cLypD7Gw.bxqry.cn
http://7iCDFz0Y.bxqry.cn
http://gtDYW08W.bxqry.cn
http://PkDFt82m.bxqry.cn
http://GCTBBFtG.bxqry.cn
http://www.dtcms.com/a/69657.html

相关文章:

  • Unity光线追踪移动端降级适配技术指南
  • Vue.js 状态管理:从本地状态到全局状态管理
  • 深入理解 Maven BOM 及其继承特性
  • 在Linux中安装Nginx
  • HarmonyOS NEXT - 电商App实例四(登录界面)
  • Unity Timeline 扩展
  • 【langchain/入门】使用langchain调用本地部署的大模型(以llama.cpp以及ollama为例)
  • maxkb安装部署
  • 企业的应用系统
  • 第四章 podman桌面版使用及Portainer UI的安装使用
  • maxwell
  • Ubuntu 24.04 Rootless Docker 安装指南
  • 动态参数二维码统计:构建APP渠道追踪体系
  • DeepSeek-R1 面试 -—— GRPO
  • 使用联核科技四向穿梭车能给企业带来哪些效益?
  • LeetCode 第4题:寻找两个正序数组的中位数
  • Linux的chmod命令,给文件设置权限
  • 【Agent实战】货物上架位置推荐助手(RAG方式+结构化prompt(CoT)+API工具结合ChatGPT4o能力Agent项目实践)
  • STC89C52单片机学习——第17节: [7-1]定时器
  • vue中常见面试题(会不断更新版)
  • 深度解读DeepSeek部署使用安全(48页PPT)(文末有下载方式)
  • 鸿蒙移动应用开发--UI组件及应用
  • Unity打包Android平台调用sherpa-onnx
  • 【VUE2】第五期——VueCli创建项目、Vuex多组件共享数据、json-server——模拟服务端api
  • MySQL隐式依赖引发的字段长度溢出:一次触发器事故的深度剖析
  • RK3588 openssl-3.4.1 编译安装
  • esProc SPL vs DuckDB:多源数据处理谁更胜一筹?
  • 编程自学指南:java程序设计开发,反射与注解,反射机制,注解
  • 【商城实战(31)】从0到1:商城项目部署全攻略
  • 提升模型准确性的关键技术与实践指南