这里以Java8为例java
首先咱们给出以下代码用来触发GCbash
public static void main(String[] args) {
// 每100毫秒建立100线程,每一个线程建立一个1M的对象,即每100ms申请100M堆空间
Executors.newScheduledThreadPool(1).scheduleAtFixedRate(() -> {
for (int i = 0; i < 100; i++) {
new Thread(() -> {
try {
// 申请1M
byte[] temp = new byte[1024 * 1024];
Thread.sleep(new Random().nextInt(1000)); // 随机睡眠1秒之内
} catch (InterruptedException e) {
e.printStackTrace();
}
}).start();
}
}, 1000, 100, TimeUnit.MILLISECONDS);
}
复制代码
咱们要模拟的场景是年轻代不断地Young GC,并有一部分对象晋升到老年代,当老年代空间不足时触发Full GC。dom
程序逻辑:每100毫秒建立100个线程,每一个线程建立一个1M的对象,即每100ms申请100M堆空间。之因此每一个线程随机睡眠1s,是为了不对象朝生夕灭,保证能够有一部分对象能晋升到老年代,更好的触发Young GC 和 Full GC,注意这个睡眠时间若是大了,会致使OOM,若是小了,很难触发FULL GC。工具
启动Java进程:java -Xms200m -Xmx200m -Xmn100m -verbose:gc -XX:+PrintGCDetails -Xloggc:./gc.log -XX:+PrintGCDateStamps -jar demo-0.0.1-SNAPSHOT.jar性能
-Xms200m -Xmx200m 最小/最大堆内存 200Mspa
-Xmn100m 年轻代内存 100M线程
-verbose:gc 开启GC日志日志
-XX:+PrintGCDetails -Xloggc:./gc.log -XX:+PrintGCDateStamps 将GC日志详情输入到gc.log中code
jcmd 获取咱们Java进程的Id:6264cdn
jmap -heap 6264查看堆信息
第一次查看,咱们发现 Eden区是98M,S0、S1是1M
第二次查看, Eden区是99M,S0、S1是0.5M
Eden区与Survivor区的比例在动态的变化,并非默认的8:1:1。
原来咱们使用默认的垃圾收集器Parallel Scavenge+Parallel Old组合,而该收集器下-XX:+UseAdaptiveSizePolicy是默认开启的,即Eden区与Survivor区比例根据GC状况会自适应变化。
咱们加上参数,关闭年轻代自适应,年轻代比例设置为8:1:1
-XX:-UseAdaptiveSizePolicy -XX:SurvivorRatio=8
另外为了尽早的触发FULL GC,咱们新增虚拟机参数
-XX:MaxTenuringThreshold=10
晋升年龄由默认的15修改成10,使得年轻代的对象更容易晋升到老年代
重启虚拟机查看jmap
Eden区80M 已使用51M,当前使用率63.8%
S0区10M 已使用0.43M,使用率4.37%
S1区10M 使用率为空
查看咱们输出的GC日志gc.log,选取其中两段
2019-06-09T02:55:30.993+0800: 330.811: [GC (Allocation Failure) [PSYoungGen: 82004K->384K(92160K)] 184303K->102715K(194560K), 0.0035647 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 2019-06-09T02:55:30.997+0800: 330.815: [Full GC (Ergonomics) [PSYoungGen: 384K->0K(92160K)] [ParOldGen: 102331K->5368K(102400K)] 102715K->5368K(194560K), [Metaspace: 16941K->16914K(1064960K)], 0.0213953 secs] [Times: user=0.02 sys=0.00, real=0.02 secs]
[GC (Allocation Failure) [PSYoungGen: 82004K->384K(92160K)] 184303K->102715K(194560K), 0.0035647 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
解释:
年轻代GC:[GC前年轻代80.08M->GC后0.37M(年轻代总大小90M)]GC前堆179.98M->GC后堆100.3M(堆总大小190M),用时]
其中年轻代总大小是90M而不是100M,这里我理解是年轻代当前最大申请到90M
100M*80%=80M 是Eden区大小
80M*80% = 64M Eden区默认占用超过8成即64M就会触发YoungGC
[Full GC (Ergonomics) [PSYoungGen: 384K->0K(92160K)] [ParOldGen: 102331K->5368K(102400K)] 102715K->5368K(194560K), [Metaspace: 16941K->16914K(1064960K)], 0.0213953 secs] [Times: user=0.02 sys=0.00, real=0.02 secs]
解释:
[GC前年轻代0.375M->GC后年轻代0M(年轻代总大小90M)][GC前老年代99.93M->GC后老年代5.24M(老年代总大小100M)]GC前堆100.3M->GC后堆5.24M(堆总大小190M),[元数据区:GC前16.5,GC后16.5(元数据区总大小1040M)],用时]
能够推测出这次FullGC缘由是年轻代晋升老年代空间不足致使
这里咱们利用 gceasy.io/ 分析一下
(1)统计年轻代、老年代、元数据区最大可用空间以及峰值,这里元数据区大小在咱们的虚拟机参数没有配置,因此取的是默认值1040M
(2)吞吐量、GC平均延迟、最大延迟以及延迟区间的统计
(3)堆所用大小的实时分析,红色位置是发生了FullGC使得堆总量直线降低
(4)GC空间总量和时间的统计
(5)各种GC时间、GC次数、GC总量等指标
GC日志分析能够帮助咱们宏观的监控GC运行状况。一方面若是频繁的FullGC会有严重的性能问题(STW),另外一方面过于频繁的GC,即GC占用系统正常运行的比重过多,吞吐量低,则是必定程度上的性能资源浪费。若系统存在性能问题,根据GC分析各项指标的做为参考,咱们也能够适当的在程序里或虚拟机参数作些调优。