我想通过代码模拟出Full GC的场景,然后记录查看Full GC日志的一些心得。首先有一些概念要提前说清楚,full gc前后一般会触发一次young gc,young gc是特指年轻代的gc,而full gc从字面理解是对整个堆内存的gc,包括年轻代、老年代、永久代。很多时候full gc很多人想表达的意思是old gc也就是老年代的gc。我这次的full gc就是old gc的意思,我想模拟对象进入老年代,然后触发old gc,看整个gc的日志。
哪些对象会进入老年代
- 长期存活躲过15次young gc的对象,也就是对象年龄达到15岁高龄会进入老年代。当然这个15岁是默认值,可以通过-XX:MaxTenuringThreshold参数设置。
- 动态年龄判断规则,如果survivor区内年龄1+年龄2+年龄n的对象所占内存大小超过了survivor区的50%,那么年龄为n的对象直接进入老年代,不一定要到15岁。
- 大对象直接进入老年代,可以通过-XX:PretenureSizeThreshold参数设置
- 如果young gc过后,存活的对象太多survivor区放不下也会进入老年代
什么时候会触发Full GC
- young gc 后进入老年代的对象所占内存总和大于老年代可用内存大小。
- 老年代空间分配担保失败,年轻代每次young gc之前,JVM都会计算下老年代剩余可用空间,如果这个可用空间小于年轻代里现有的所有对象大小之和(包括垃圾对象)。就会看看是否设置了“-XX:-HandlePromotionFailure”(jdk1.8默认就设置了)参数,如果有这个参数,就会看看老年代的可用内存大小,是否大于之前每一次minor gc后进入老年代的对象