每个垃圾收集器从以下三点总结:
1.使用的垃圾收集算法
2.工作在新生代还是老年代
3.特定与工作过程(关键的JVM参数)
serial收集器
1.新生代使用标记复制算法、老年代使用标记整理算法。
2.新生代和老年代。
3.只有一个线程去收集,收集过程会停止用户线程(STW)。适合于单CPU的计算机。
-XX:SurvivorRatio :Eden区与SruvivorRatio的比例,注意survivor有两个哈。
-XX:PretenureSizeThreshold:新生代对象晋升老年代的年龄。
ParNew收集器
1.使用标记复制
2.工作在新生代
3.使用多个线程去收集,可以配合CMS工作。也会(STW)。
开启和CPU相同个数线程去收集垃圾。
可以用下面参数去限制使用的线程数
-XX:ParallelGCThreads
Parallel Scavenge
1.标记复制算法。
2.工作在新生代。
3.基本概念与ParNew相同,重点关注吞吐量。而其他收集器大多关注减少暂停时间。
吞吐量 = (运行用户代码的时间) / (运行用户代码的时间 + 运行垃圾收集代码的时间)。
既吞吐量,关注 减少垃圾收集线程的运行时间。
看起来像是提高吞吐量和减少暂停时间是一个意思,其实不然。
减少暂停时间,一般是使用并发技术,让用户线程不暂停。但这样,由于线程切换,可能垃圾收集线程运行的时间要比,暂停用户线程,去运行的时间长。
两个参数:
-XX:MaxGCPauseMillis:最长停顿时间。
-XX:GCTimeRatio:吞吐量大小。
-XX:+UseAdaptiveSizePolicy:让收集器根据系统情况,自动设置参数(SurvivorRatio等),来提供最大的吞吐量。
上面这个自适应调节策略也是与ParNew收集器的一个重要区别。
Parallel Old
1.使用标记整理
2.老年代的Parallel
3.吞吐量优先
CMS Concurrent Mark Sweep
1.标记-清除
2.老年代
3.整个过程拢共分为五个步骤,仔细看一下:
a.初始标记,标记GC Root可以直接引用到的对象,需要STW。
b.并发标记,根据初始标记的节点们,进行可达性分析。注意这个时候是和用户线程并发执行的。这时候就有两个问题。第一个问题:如果在并发收集的过程中,用户新创建了对象,那么这个新创建的对象,不在初始标记的链中,不是要被删除了么?怎么办,所以下面有重新标记。第二个问题,如果在并发标记的过程中,原来已经标记的对象不用了,怎么办?这个叫做浮动垃圾,留到下一次GC时收集。
c.重新标记,需要STW。既解决上面说的第一个问题。
d.并发清除。
CMS的缺点:
1.CPU敏感。占用CPU导致用户线程变慢。
2.无法处理浮动垃圾,无法处理并发过程产生的对象垃圾。
3.是标记-清除算法的,有碎片,会在老年代有空间的情况下提前进入GC。有个参数
-XX:+UseCMSCompactAtFullCollection,在fullGC的是时候,压缩一手。这样停顿时间会变长。
CMS的宗旨是,减少停顿时间。(不是没有蛀牙哈)
G1垃圾收集器
1.将java堆分为独立的区域Region,保留新生代,老年代的概念。
2.特点:可预测的停顿。那么如何实现可预测的停顿呢?G1跟踪各个Region里面的垃圾堆的价值大小(分析,回收空间大小以及回收所需时间的大小)。
- G1是一个有整理内存过程的垃圾收集器,不会产生很多碎片。
- G1的STW可控,给停顿时间加上了预测机制。
原来的收集器,内存,新生代,老年代是连续的。
G1,是不连续的。
预测机制:会根据允许收集的时间,选择回收价值最大的Region,使用Region划分和优先级区域的回收方式,保证了G1可以在有限的时间获取尽可能高的收集效率。
网友评论