Java运行时内存
Java 堆从GC 的角度还可以细分为: 新生代(Eden
区、From Survivor
区和To Survivor
区)和老年
代。
一、新生代
是用来存放新生的对象。一般占据堆的1/3
空间。由于频繁创建对象,所以新生代会频繁触发
MinorGC
进行垃圾回收。新生代又分为Eden
区、ServivorFrom
、ServivorTo
三个区。
-
Eden
Java 新对象的出生地(如果新创建的对象占用内存很大,则直接分配到老
年代)。当 Eden 区内存不够的时候就会触发 MinorGC,对新生代区进行
一次垃圾回收。 -
ServivorFrom
上一次 GC 的幸存者,作为这一次 GC 的被扫描者。
-
ServivorTo
保留了一次 MinorGC 过程中的幸存者。
-
MinorGC的过程(复制->清空->互换)
-
step1:eden、servicorFrom 复制到 ServicorTo,年龄+1
首先,把 Eden 和 ServivorFrom 区域中存活的对象复制到 ServicorTo 区域(如果有对象的年
龄以及达到了老年的标准,则赋值到老年代区),同时把这些对象的年龄+1(如果 ServicorTo 不
够位置了就放到老年区); -
step2: 清空 eden、servicorFrom
然后,清空 Eden 和 ServicorFrom 中的对象;
-
step3:ServicorTo 和 ServicorFrom 互换
最后,ServicorTo 和 ServicorFrom 互换,原 ServicorTo 成为下一次 GC 时的 ServicorFrom
区。
-
二、老年代
- 主要存放应用程序中生命周期长的内存对象。
- 老年代的对象比较稳定,所以
MajorGC
不会频繁执行。在进行MajorGC
前一般都先进行
了一次MinorGC
,使得有新生代的对象晋身入老年代,导致空间不够用时才触发。当无法找到足
够大的连续空间分配给新创建的较大对象时也会提前触发一次MajorGC
进行垃圾回收腾出空间。 MajorGC
采用标记清除算法:首先扫描一次所有老年代,标记出存活的对象,然后回收没
有标记的对象。MajorGC
的耗时比较长,因为要扫描再回收。MajorGC
会产生内存碎片,为了减
少内存损耗,我们一般需要进行合并或者标记出来方便下次直接分配。当老年代也满了装不下的
时候,就会抛出OOM
(Out of Memory
)异常。
三、永久代
指内存的永久保存区域,主要存放 Class
和Meta
(元数据)的信息,Class
在被加载的时候被
放入永久区域,它和和存放实例的区域不同,GC
不会在主程序运行期对永久区域进行清理。所以这
,Class
在被加载的时候被
放入永久区域,它和和存放实例的区域不同,GC
不会在主程序运行期对永久区域进行清理。所以这
也导致了永久代的区域会随着加载的 Class
的增多而胀满,最终抛出 OOM
异常。