JVM GC机制

 垃圾收集主要是针对堆和方法区进行。

回收机制:

  现在的JVM基本都使用分代回收机制,把堆中内存区域分为新生代,老年代。

  新生代:

Eden(80%) Survivor0(10%) Survivor1(10%)

  老年代:

Old place

  

  回收算法:

    Minor GC:发生在新生代,内存空间小,频繁执行,所以一般执行速度很快。

    Full GC(Major GC):Major GC发生在老年代,一般会伴随着一次Minor GC,所以也叫Full GC,一般执行速度较慢。

    

    在新生代中采用复制算法,每个新对象都最先在Eden里,当Eden满了之后(此时两个Survivor都为空),将触发一次Minor GC,Eden里存活的对象就会送到Survivor0中,即 Eden→Survivor0,Eden里对象清空。

  下一次Eden区满之后,就将Eden和Survivor0中存活的对象放到Survivor1中,即Eden+Survivor0→Survivor1。下次Eden满了之后,Eden+Survivor1→Survivor0..........................

  如此循环,当对象的复制次数达到16次时,对象就会被送到老年代。

    当新生代中对象放入老年代,而老年代中空间不足时,将会进行Full GC。

    

为什么需要Survivor,又为什么需要两个呢?

  先假设没有Survivor:

    Eden中的对象经历Minor GC后送到Old place,因为Minor GC很频繁,那么Old place很快就被填满了,然后进行Full GC,然而Full GC很消耗时间,会影响程序的执行速度和响应速度。

    若增大Old place的空间,虽然Full GC的频率降低了,但是每次Full GC的时间增长了。若减小Old place的空间,Full GC的时间减少,但是频率增加了。

    由此可见,增加Survivor的作用是,减少送往老年代的对象,从而降低Full GC的频率。

  再假设只有一个Survivor:

    Eden中的对象经历Minor GC后送到空的Survivor,此时没有什么问题,再然后又经历Minor GC,此时Survivor中有些对象死亡,而新的对象从Eden中过来,就出现了碎片化的问题。

    

    由此可见,加一个Survivor是不可取的,至少的加两个,那为什么不三个四个呢?我觉得可能是因为新生代和老年代的空间大小比例有关,若增加多了Survivor,则会降低老年代的利用率,所以选择了两个Survivor。

    所以设置两个Survivor的原因就是,解决碎片化。

回收器:

  新生代:

      Serial回收器:

        串行回收器,Client 模式下的默认新生代收集器,是一个单线程的回收器,所以对于只有一个CPU来说,利用率最高。

      ParNew回收器:

        并行回收器,Server 模式下的虚拟机首选新生代收集器,能与CMS回收器配合,是多线程版的Serial,默认开启线程数量与CPU核数相同,可以使用 -XX:ParallelGCThreads 参数来设置线程数。

      Parallel Scavenge回收器:

        同样是多线程,但是其余回收器都是以缩短“垃圾回收时间”为主,而Parallel Scavenge的目标是达到一个可控制的吞吐量,它被称为“吞吐量优先”收集器。这里的吞吐量指 CPU 用于运行用户代码的时间占总时间的比值。

        停顿时间越短就越适合需要与用户交互的程序,良好的响应速度能提升用户体验。而高吞吐量则可以高效率地利用 CPU 时间,尽快完成程序的运算任务,适合在后台运算而不需要太多交互的任务。

  老年代:

      Serial Old回收器:

        同Serial回收器。

      Parallel Old回收器:

        同Parallel Scavenge回收器。

      CMS回收器:

        CMS是Concurrent(同时发生的)- Mark(标记)- Sweep(清理),所以此回收器采用标记-清理算法。分为以下四个步骤:  

        1、初始标记:仅仅只是标记一下 GC Roots 能直接关联到的对象,速度很快,需要停顿。

        2、并发标记:进行 GC Roots Tracing 的过程,它在整个回收过程中耗时最长,不需要停顿。

        3、重新标记:为了修正并发标记期间因用户程序继续运作而导致标记产生变动的那一部分对象的标记记录,需要停顿。

        4、并发清除:不需要停顿。

        在整个过程中耗时最长的并发标记和并发清除过程中,收集器线程都可以与用户线程一起工作,不需要进行停顿。

  G1回收器:

    G1(Garbage-First),它是一款面向服务端应用的垃圾收集器,在多 CPU 和大内存的场景下有很好的性能。HotSpot 开发团队赋予它的使命是未来可以替换掉 CMS 收集器。

  堆被分为新生代和老年代,其它收集器进行收集的范围都是整个新生代或者老年代,而 G1 可以直接对新生代和老年代一起回收。

   参考资料:

      1、JVM GC 机制与性能优化

      2、为什么新生代内存需要有两个Survivor区

      3、Java虚拟机

原文地址:https://www.cnblogs.com/zhuii/p/9961764.html