我们知道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
list = new ArrayList(); - 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抛出内存溢出异常.