java对象从新生代晋升到老年代的四种方式

Posted java叶新东老师

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了java对象从新生代晋升到老年代的四种方式相关的知识,希望对你有一定的参考价值。

晋升的四种方式

  1. 担保机制
  2. 大对象直接进入老年代
  3. 长期存活的对象
  4. 动态年龄判断

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 空间的一半,年龄大于或等于该年龄的对象就可以直接进入老年代,而不需要达到默认的分代年龄。

以上是关于java对象从新生代晋升到老年代的四种方式的主要内容,如果未能解决你的问题,请参考以下文章

GC新生代对象晋升到老年代情况总结

收藏:

java 测试对象动态判断是否需要晋升到老年代

java 测试长期存活的对象将晋升到老年代

CMS 垃圾收集流程

JVM 晋升到老年代的动态年龄判断