晋升的四种方式
- 担保机制
- 大对象直接进入老年代
- 长期存活的对象
- 动态年龄判断
1、担保机制
1.1什么是担保机制
我们都用过花呗借呗、或者信用卡吧? 就是你身上的前不够用的时候,你可以去借信用卡、借花呗、借借呗,但是你必须有个东西来担保,不然人家凭什么借给你?现在来说一般都是用你的信用分来担保的,就像支付宝有芝麻信用分数;还有一些比较大的数目是需要你用车子或房子来抵押的,这就是担保,以确保你不会跑掉;那其实jvm在内存分配的时候也有担保机制,就是你的新生代内存不足的时候,通过担保分配的方式让大对象直接分配到老年代;
1.2 代码示例
接下来,我们运行一组代码测试一下,在运行前需要先加上以下几个jvm的参数
- -XX:+PrintGCDetails -verbose:gc -Xms30M -Xmx30M
-
- 参数说明
- * -XX:+PrintGCDetails 程序执行完成后打印内存使用情况
- * -verbose:gc 每次GC后打印日志
- * -Xms30M 初始堆内存
- * -Xmx30M 最大堆内存
java 代码 GcTest.java
- public class GcTest {
-
- // 占用1MB空间
- private static final int _1MD = 1024 * 1024;
-
- public static void main(String[] args) {
- byte[] bytes = new byte[2 *_1MD];
- byte[] bytes1 = new byte[2 * _1MD];
- byte[] bytes2 = new byte[2 * _1MD];
- byte[] bytes3 = new byte[6 * _1MD];
- }
-
- }
运行后结果如下
由上面的结果可以看到,前面的三个 2M 的对象都分配到了eden (伊甸)区,此时eden区的内存占用已经达到了 99%,第四个占用了6M的对象因为eden区已经放不下了,所以通过了担保机制直接分配到了 ParOldGen (老年代 ) ;
2、大对象直接进入老年代
2.1 多大的对象才能称为大对象?
这个大对象的大小是由用户指定的,使用以下jvm参数进行指定
-XX:PretenureSizeThreshold=对象大小(单位:byte)
这个参数的默认值为0,也就是说,所有的对象创建出来之后默认都是分配到新生代的,当我们指定了大小之后,只要创建出来的对象超过设定值,那么这个对像就会直接晋升到老年代;
需要注意的是:PretenureSizeThreshold参数只对Serial和ParNew两款收集器有效,
2.2 代码示例
首先,加上以下几个参数
- -XX:+PrintGCDetails -verbose:gc -Xms30M -Xmx30M -XX:+UseParNewGC -XX:PretenureSizeThreshold=3145728
-
- 参数说明
- * -XX:+PrintGCDetails 程序执行完成后打印内存使用情况
- * -verbose:gc 每次GC后打印日志
- * -Xms30M 初始堆内存
- * -Xmx30M 最大堆内存
- * -XX:+UseParNewGC 使用parNew GC
- * -XX:PretenureSizeThreshold=3145728 对象超过3M时直接晋升到老年代
java代码,这里将对象设为4M,
- public class GcTest {
-
- private static final int _1MD = 1024 * 1024;
-
- public static void main(String[] args) {
- byte[] bytes = new byte[4 *_1MD];
- }
-
- }
打印结果
由上面的结果可以看到,因为对象的大小设置为4M,已经超过了参数 PretenureSizeThreshold 设置值3M,所以直接将该对象分配到了老年代;
3、长期存活的对象
长期存活的对象进入老年代。在堆中分配内存的对象,其内存布局的对象头中(Header)包含了 GC 分代年龄标记信息。如果对象在 eden 区出生,那么它的 GC 分代年龄会初始值为 1,每熬过一次 Minor GC 而不被回收,这个值就会增加 1 岁。当它的年龄到达一定的数值时(jdk1.7 默认是 15 岁),就会晋升到老年代中。
4、动态年龄判断
动态对象年龄判定。当 Survivor 空间中相同年龄所有对象的大小总和大于 Survivor 空间的一半,年龄大于或等于该年龄的对象就可以直接进入老年代,而不需要达到默认的分代年龄。