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

发布时间:2022-03-01 11:18:46 作者:yexindonglai@163.com 阅读(602)

晋升的四种方式

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

1、担保机制

1.1什么是担保机制

     我们都用过花呗借呗、或者信用卡吧? 就是你身上的前不够用的时候,你可以去借信用卡、借花呗、借借呗,但是你必须有个东西来担保,不然人家凭什么借给你?现在来说一般都是用你的信用分来担保的,就像支付宝有芝麻信用分数;还有一些比较大的数目是需要你用车子或房子来抵押的,这就是担保,以确保你不会跑掉;那其实jvm在内存分配的时候也有担保机制,就是你的新生代内存不足的时候,通过担保分配的方式让大对象直接分配到老年代;

1.2 代码示例

接下来,我们运行一组代码测试一下,在运行前需要先加上以下几个jvm的参数

  1. -XX:+PrintGCDetails -verbose:gc -Xms30M -Xmx30M
  2. 参数说明
  3. * -XX:+PrintGCDetails 程序执行完成后打印内存使用情况
  4. * -verbose:gc 每次GC后打印日志
  5. * -Xms30M 初始堆内存
  6. * -Xmx30M 最大堆内存

java 代码 GcTest.java

  1. public class GcTest {
  2. // 占用1MB空间
  3. private static final int _1MD = 1024 * 1024;
  4. public static void main(String[] args) {
  5. byte[] bytes = new byte[2 *_1MD];
  6. byte[] bytes1 = new byte[2 * _1MD];
  7. byte[] bytes2 = new byte[2 * _1MD];
  8. byte[] bytes3 = new byte[6 * _1MD];
  9. }
  10. }

运行后结果如下

由上面的结果可以看到,前面的三个 2M 的对象都分配到了eden (伊甸)区,此时eden区的内存占用已经达到了 99%,第四个占用了6M的对象因为eden区已经放不下了,所以通过了担保机制直接分配到了 ParOldGen (老年代 ) ;

 

2、大对象直接进入老年代

2.1 多大的对象才能称为大对象?

    这个大对象的大小是由用户指定的,使用以下jvm参数进行指定

-XX:PretenureSizeThreshold=对象大小(单位:byte)

这个参数的默认值为0,也就是说,所有的对象创建出来之后默认都是分配到新生代的,当我们指定了大小之后,只要创建出来的对象超过设定值,那么这个对像就会直接晋升到老年代;

需要注意的是:PretenureSizeThreshold参数只对Serial和ParNew两款收集器有效,

2.2 代码示例

首先,加上以下几个参数

  1. -XX:+PrintGCDetails -verbose:gc -Xms30M -Xmx30M -XX:+UseParNewGC -XX:PretenureSizeThreshold=3145728
  2. 参数说明
  3. * -XX:+PrintGCDetails 程序执行完成后打印内存使用情况
  4. * -verbose:gc 每次GC后打印日志
  5. * -Xms30M 初始堆内存
  6. * -Xmx30M 最大堆内存
  7. * -XX:+UseParNewGC 使用parNew GC
  8. * -XX:PretenureSizeThreshold=3145728 对象超过3M时直接晋升到老年代

java代码,这里将对象设为4M,

  1. public class GcTest {
  2. private static final int _1MD = 1024 * 1024;
  3. public static void main(String[] args) {
  4. byte[] bytes = new byte[4 *_1MD];
  5. }
  6. }

打印结果

由上面的结果可以看到,因为对象的大小设置为4M,已经超过了参数 PretenureSizeThreshold 设置值3M,所以直接将该对象分配到了老年代;

 

3、长期存活的对象

长期存活的对象进入老年代。在堆中分配内存的对象,其内存布局的对象头中(Header)包含了 GC 分代年龄标记信息。如果对象在 eden 区出生,那么它的 GC 分代年龄会初始值为 1,每熬过一次 Minor GC 而不被回收,这个值就会增加 1 岁。当它的年龄到达一定的数值时(jdk1.7 默认是 15 岁),就会晋升到老年代中。

 

4、动态年龄判断

动态对象年龄判定。当 Survivor 空间中相同年龄所有对象的大小总和大于 Survivor 空间的一半,年龄大于或等于该年龄的对象就可以直接进入老年代,而不需要达到默认的分代年龄。

关键字Java