图解Java中的GC(分代收集器)

前面在Java垃圾收集算法中讲过垃圾收集算法中的分代收集器,今天看了一个视频发现里面将的也很不错,所以决定再总结一下。

我们知道,在分代收集算法中堆空间被分为新生代和老年代。因为新生代中对象的存活率比较低,所以一般采用复制算法,老年代的存活率一般比较高,一般使用”标记-清理”或者”标记-整理”算法进行回收。

图解Java中的GC(分代收集器)_第1张图片


上面的这个图已经很清楚的将堆的分区展现出来了。


下面我们来看看具体的算法过程。

新创建的对象一般放在新生代的Enden区,如下图所示。

图解Java中的GC(分代收集器)_第2张图片


上面对象中,绿色代表的是"存活对象",灰色的代表的是"待回收对象"。当Enden中被使用完的时候,就会发生新生代GC,也就是Minor GC,如下图。

图解Java中的GC(分代收集器)_第3张图片


首先会把存活对象复制到Survivor1中。

图解Java中的GC(分代收集器)_第4张图片


然后把Enden清空

图解Java中的GC(分代收集器)_第5张图片


移动到Survivor1空间后,设置对象年龄(Age)为1

图解Java中的GC(分代收集器)_第6张图片

这样第一次GC就完成了。


当Enden区再次被使用完的时候,就会再次进行GC操作

图解Java中的GC(分代收集器)_第7张图片

上面Enden和Survivor1中,绿色表示存活对象,回收表示"待回收对象",因为在堆内存使用分配的过程中,也会不断有对象变得引用不可达。


再次GC的过程中,跟上面一样,将Enden区和Survivor1中的存活对象复制到Survivor2中。需要注意的是目前还是处于新生代的GC,因为新生代分为Enden、Survivor1、Survivor2三个区,使用的其实就是复制算法。

图解Java中的GC(分代收集器)_第8张图片


接着将Enden和Survivor1进行清空

图解Java中的GC(分代收集器)_第9张图片


然后将Enden中复制到Survivor2中的对象年龄设置为1,将Survivor1中复制到Survivor2中的对象年龄加1

图解Java中的GC(分代收集器)_第10张图片


这样新生代第二次GC就完成了。当Enden再一次被使用完的时候,就会发生第三次GC操作了。

图解Java中的GC(分代收集器)_第11张图片


下面基本重复上面的思路了,首先将Enden和Survivor2中的存活对象复制到Survivor1中。

图解Java中的GC(分代收集器)_第12张图片


然后将Enden和Survivor2进行清空

图解Java中的GC(分代收集器)_第13张图片


然后将Enden中复制到Survivor1中的对象年龄设置为1,将Survivor2中复制到Survivor1中的对象年龄加1

图解Java中的GC(分代收集器)_第14张图片


后面的操作基本都是重复的,那什么时候会进入老年代呢?从上面看到,如果对象在GC过程中没有被回收,那么它的对象年龄(Age)会不断的增加,对象在Survivor区每熬过一个Minor GC,年龄就增加1岁,当它的年龄到达一定的程度(默认为15岁),就会被移动到老年代,这个年龄阀值可以通过-XX:MaxTenuringThreshold设置。

图解Java中的GC(分代收集器)_第15张图片


视频链接:Garbage collection in Java, with Animation and discussion of G1 GC


你可能感兴趣的:(Java虚拟机)