0


创建对象在Heap堆区中如何分配内存

前言

    我们知道Heap堆区是JVM所管理的内存最大并且内存被所有线程共享的一块区域,Heap堆区是用来存放对象实例和数组的,那Heap堆区又是如何对创建的对象进行内存分配的呢?在这里我们就要提到两个词“新生代”和“老年代”

新生代、老年代

    Heap堆是垃圾收集器(Garbage Collected)管理的主要区域,因此堆区又被称为GC堆,在JVM中堆区往往进行分带划分,所有堆区又分为新生代和老年代,目的是为了更高效的回收与分配内存

    Heap堆区可以在命令提示符通过java -xx:+PrintFlagsFinal -verstom命令查看新生代与老年代的空间分配比例:

8

     由 InitialSurvivorRatio=8 可知新生代Young(Eden/Survivor)空间的初始比例为8;代表Eden占新生代空间的80%

      由 uintx NewRatio=2 可知老年代Old/新生代Young空间的比例为2;代表老年代Old是新生代Young的2倍

综上分析可得如下图:

创建对象的内存分配

我们可以通过这段代码观察Heap堆区创建对象的内存分配过程:

public class TestOOM {
      static String base = "aabbcc";
      public static void main(String[] args) {
        List<String> list = new ArrayList<String>();
        for (int i=0;i< Integer.MAX_VALUE;i++){
          String str = base + base;   //堆区
          base = str;
          list.add(str.intern());    //字符串常量池
        }
      }
    }

输出结果:

创建一个新对象,在堆中的分配内存。我们可以通过如下表概述对象的内存分配过程:

总结:

       当创建一个对象时,对象首先会检查在 
Eden

是否有足够存放空间,当

Eden

区存放空间足够时,就会给当前对象分配内存空间,当

Eden

区存放空间不足时,这时会触发

Young Garbage Collection

,即

YGC

垃圾回收,在

Eden

区实现清除策略,没有被引用的对象则直接回收。而依然存活的对象会被移送到

Survivor

区。

Survivor

区分为 from(S0) 和** to** (S1)两块内存区域。每次

YGC

的时候,它们将存活的对象复制到未使用的

Survivor

空间(

**s0**

** 或 **

**s1**

),然后将当前正在使用的空间完全清除,交换两块空间的使用状态。每次交换时,对象的年龄会加

+1

。如果

YGC

要移送的对象大于

Survivor

区容量的上限,则直接移交给老年代。一个对象也不可能永远呆在新生代,在

JVM

中 一个对象从新生代晋升到老年代的阈值默认值是

15

,可以在

Survivor

区交换 14 次之后,晋升至老年代。

           当YGC之后,又会检查在 
Eden

是否有足够存放空间当前对象,如果内存空间足够,则在Eden区分配内存空间,如果放不下则会检查Old区是否有足够的存放空间,如果有,则在老年代分配内存空间,否则就会进行FGC垃圾回收,之后再一次进行检查Old区是否有足够的存放空间,如果有,则在老年代分配内存空间,否则OOM抛出内存溢出异常.

标签: jvm java 开发语言

本文转载自: https://blog.csdn.net/w259149/article/details/126919453
版权归原作者 雾远望 所有, 如有侵权,请联系我们删除。

“创建对象在Heap堆区中如何分配内存”的评论:

还没有评论