扩容新生代为什么能够提高GC的效率

扩容新生代为什么能够提高GC的效率

该文章默认读者对JVM的基础有所了解

在学习JVM的时候,遇到了个人感觉比较有意思的问题,通过视频学习整理了一下。

先来上图:

image-20220125151642305

大部分情况下,对象都会进入Eden区进行分配,当该空间满了以后,进行轻GC,如果对象存活,则会进入s0或者s1。

扩展:如何判断对象存活

采用根可达性分析算法来判断

可作为根的对象有:

  1. 虚拟机栈中引用的对象
  2. 本地方法栈中引用的对象
  3. 方法区中类静态属性引用的对象
  4. 方法区常量引用的对象

GC的耗时由两部分组成:

  1. 扫描整个新生代所要花费的时间
  2. 将存活对象复制到幸存区的时间

所以CG耗时=T1+T2;

假设:

新生代空间 (Space1): 200m,每隔5s GC一次,每次GC耗时100ms

GC=T1+T2=100ms

扩容新生代空间(Space2)(两倍): 400m,每隔10s GC一次,每次GC耗时200ms

GC=2*(T1+T2) =200ms ?(对不对)

其中我们需要考虑对象A的存活时间,它会影响到T2时间。

假设对象A存活时间:7S;

在Space1中 GC的时候对象A是存活的,那么就需要对对象进行移动,所以Space1中的GC=T1+T2。

在Space2中GC的时候对象A变为垃圾,那么就不需要对垃圾进行移动,所以在Space2中的GC=2*T1;

在垃圾回收算法中,JVM中移动内存对象的耗时比扫描(算法有优化)整个新生代的耗时是远远高于的。(前提条件)

由此可以得出,扩大新生代确实能够提高GC的效率,其中起决定的因素不是空间的大小,而是存活对象的大小;

增加整个空间的大小,可以拉长GC的间隔,从而尽可能的使得对象变为垃圾,来减少内存的移动。