美文网首页
怎么让 JVM 几乎不发生 Full GC?

怎么让 JVM 几乎不发生 Full GC?

作者: wyn_做自己 | 来源:发表于2021-12-05 15:38 被阅读0次

今日份鸡汤:当你快挺不住的时候,磨难也挺不住了~

先来看一个业务流程图:

image.png

问题描述:

(1)一个 4 核 8G 的订单系统,假设给 JVM 运行内存为 3 个G,根据堆内存划分比例,老年代可分 2G,Eden 800M,S0/S1 各 100M。
(2)线程运行每秒产生 60M 对象,大概运行 13 秒就会占满 Eden 区,前 12 秒产生的对象在做一个 minorgc 后被当作垃圾对象处理掉,第 13 秒产生的对象不是垃圾对象,会被放到 S0 区。
(3)第 13 秒产生的 60M 对象由于大于 S0 区的 50% 所以会被放到老年代。为了能更好地适应不同程序的内存状况,HotSpot虚拟机并不是永远要求对象的年龄必须达到 -XX:MaxTenuringThreshold 才能晋升老年代,如果在Survivor空间中相同年龄所有对象大小的总和大于Survivor空间的一半,年龄大于或等于该年龄的对象就可以直接进入老年代,无须等到 -XX:MaxTenuringThreshold 中要求的年龄。
因此每隔 13 秒就有 60M 对象会被放到老年代,大概 7 到 8 分钟就会放满老年代,老年代放满后就会产生一次 full gc,此时老年代里 99% 的对象是垃圾对象,会被清理掉。而一次 full gc,会收集整个堆的垃圾对象,时间过长。因此系统每隔七八分钟就会有持续性的卡顿现象

问题解决:

Full GC 最根本的产生原因就是有对象不停的进入老年代,最后导致空间不足,引发 Full GC。解决思路就是直接破坏掉产生条件,直接减少运行时期间从新生代晋升到老年代的对象,或者没有对象晋升到老年代就行了。具体措施:

(1)大对象频繁进行老年代,造成老年代空间快速被占满,造成 Full GC。
解决:合理配置-XX:PretenureSizeThreshold大小。避免过多非必要对象进入老年代。

(2)metaspace 空间不足
解决:一般这个里面存放的都是一些 Class 类信息,Class 本身也是一个对象,需要空间存放。那么程序代码中什么时候会产生对象进入呢,当使用 CGLIB 动态代理不停的生成代理类的时候,就会加载到元数据空间,当然一般 4 核 8G 内存的物理机分配个 512M 是完全没问题的。

(3)从年轻代晋升到老年代的对象
长期存活对象:达到了设置的年龄限制,默认是 15 次。Young GC 后,存活对象大于 survivor 区,存活对象全部进入老年代,注意动态年龄的区别。
动态年龄判定:如果在 Survivor 空间中相同年龄所有对象大小的总和大于Survivor 空间的一半,年龄大于或等于该年龄的对象就可以直接进入老年代。
<meta charset="utf-8">

解决:

1> 针对"长期存活对象"调大晋升年龄没有多大意义。假设 10 秒一次 Young GC 15 * 10 = 150s,存活两分钟的对象,可以认为是系统中长期存活的对象,调大一点,也仅仅是让它在新生代在多待一会儿,还不如让它早点去它该区的老年区。

2> 存活对象大于存活区大小和"动态年龄判定"二者产生原因差不多,都是 Survivor 区大小分配不合理,可以同时进行优化。核心思想就是合理分配新生代老年代内存比例大小。


image.png

如图调整内存比例,线程运行每秒产生 60M 对象,大概运行 28 秒就会占满 Eden 区,此时前 27 秒产生的对象在做一个 minor gc 后被当作垃圾对象销毁掉,第 28 秒产生的对象会被放到 S0 区,由于 60M 小于 S0 区的 50% 不会被放到老年代。当 Eden 再一次放满,此时 minor gc 会销毁 Eden 中前 27 秒的垃圾对象和 S0 中的对象,Eden 第 28 秒产生的对象会被放到 S1 区。当 Eden 再一次放满,minor gc 会销毁 Eden 中前 27 秒的垃圾对象和 S1 中的对象,Eden 第 28 秒产生的对象会被放到 S0 区,如此 JVM 几乎不发生 full gc。

调优:线上如何合理的去分配内存?
(1)找到最大的压力瓶颈点,也可以说并发最多的点。

(2)根据系统未来的业务量,访问量,去推算这个系统每秒的并发量,注意项目的特殊性,一般可以用二八原则。

(3)计算一次业务新生代会占用多少内存,一般可以考虑主要业务对象大小扩大10~20倍来预估,或者直接用工具直接监测。

(4)一次业务的时长,即一次请求的耗时。

(5)根据 1 和 2 中推算的每秒的并发量预估推算对内存空间的占用量,这个占用量是一秒内或者说在并发过程中无法回收的。无法回收的对象大小 = 业务处理时间 * QPS * 每个处理会产生的对象大小

(6)根据内存的推算结果预估出运行期间 JVM 的内存运转模型。

(7)部署多少台机器,每台机器配置如何。

相关文章

  • 怎么让 JVM 几乎不发生 Full GC?

    今日份鸡汤:当你快挺不住的时候,磨难也挺不住了~ 先来看一个业务流程图: 问题描述: (1)一个 4 核 8G 的...

  • 如何让 JVM 几乎不发生 full gc

    一、业务流程简图 二、问题分析 1️⃣一个 4 核 8G 的订单系统,假设给 JVM 运行内存为 3 个G,根据堆...

  • 4

    1. Jvm中的Full GC是什么?Full GC发生的场景?如何应对? Full GC讲解之前,先看一张JVM...

  • JVM垃圾回收日志各字段详解

    JVM垃圾回收日志各字段详解。 一、演示Full GC [Full GC (Ergonomics) [PSYoun...

  • JVM性能调优的6大步骤,及关键调优参数详解

    JVM内存调优 对JVM内存的系统级的调优主要的目的是减少GC的频率和Full GC的次数。 1.Full GC ...

  • Java架构师面试题——JVM性能调优

    JVM内存调优 对JVM内存的系统级的调优主要的目的是减少GC的频率和Full GC的次数。 1.Full GC ...

  • JVM调优

    一、JVM内存调优 对JVM内存的系统级的调优策略主要是减少GC的频率和Full GC的次数。 1️⃣Full G...

  • jvm-doc

    触发JVM进行Full GC的情况及应对策略

  • JVM调优6大步骤

    对JVM内存的系统级的调优主要的目的是减少GC的频率和Full GC的次数。 1.Full GC 会对整个堆进行整...

  • JVM性能调优的六大步骤

    对JVM内存的系统级的调优主要的目的是减少GC的频率和Full GC的次数。 1.Full GC 会对整个堆进行整...

网友评论

      本文标题:怎么让 JVM 几乎不发生 Full GC?

      本文链接:https://www.haomeiwen.com/subject/dlnixrtx.html