浅谈Java堆内存分代回收
浅谈Java堆内存分代回收
概述
与C++不同的是:在Java中我们无需关心对象占用空间的释放,这主要得益于Java中的垃圾处理器(简称GC)帮助我们自动的进行对象占用空间的释放。
下面我们带着几个问题来学习:
- 堆内存是如何分代的?
- 各分代之间是如何配合工作的?
1、堆内存是如何分代的?
用一张图片来描述(面积大小不代表实际占用空间大小)
堆内存分为:年轻代(Young) + 老年代(Old),年轻代又分为:Eden区 + Survivor区 * 2。
通常年轻代中的各区比值为:Eden区:Survivor0 :Survivor1 = 8:1:1
要尽可能的让对象不进入Old区。
S0和S1默认情况下会动态的自动调整大小,可以使用`-XX:UseAdaptiveSizePolicy`来关闭动态调整
2、各分代之间是如何配合工作的?
以一个对象的在每个区之间的复制来描述这个问题,我们假定这个对象一直存活着。
当我们写一段如下代码时
User user = new User();
user对象首先被放入到Eden区,当Eden区满的时候会发生第一次Minor GC,这时垃圾收集器会在S0和S1中随机选择一个区(假设选中了S0)来存放Eden区剩余存活的对象。当Eden区再次满的时候会发生第二次Minor GC,这时垃圾收集器会把Eden区存活的对象 + S0中存活的对象复制到S1中,当Eden区再次满的时候会发生第三次Minor GC,这时垃圾收集器会把Eden区存活的对象 + S1中存活的对象复制到S0中,如此循环复制。
那么对象什么情况下会进入到Old区?
- Eden区满时,在对象将要进入S0或S1中时,如果S0或S1存放不下Eden区中某个对象时,这个对象将被复制到Old区。
- 如果一个对象在S0和S1中经历了制定次数的复制之后,会被复制到Old区。
顺便说一下:年轻代垃圾回收使用复制算法,老年代回收使用标记清除算法。