jvm(8) -- Serial收集器、ParNew收集器、Parallel Scavenge收集器

简介: jvm(8) -- Serial收集器、ParNew收集器、Parallel Scavenge收集器

1.Serial收集器


推荐文章:https://www.breakyizhan.com/javamianshiti/2850.html


Serial,是单线程执行垃圾回收的。当需要执行垃圾回收时,程序会暂停一切手上的工作,然后单线程执行垃圾回收。


因为新生代的特点是对象存活率低,所以收集算法用的是复制算法,把新生代存活对象复制到老年代,复制的内容不多,性能较好。


Serial(串行)垃圾收集器是最基本、发展历史最悠久的收集器;


JDK1.3.1前是HotSpot新生代收集的唯一选择;


①特点


针对新生代;


采用复制算法;


单线程收集;


进行垃圾收集时,必须暂停所有工作线程,直到完成;


即会"Stop The World";


Serial/Serial Old组合收集器运行示意图如下:

1dc618a0ed9580ce8bfa6facb208c08f.png


②、应用场景


依然是HotSpot在Client模式下默认的新生代收集器;


也有优于其他收集器的地方:

简单高效(与其他收集器的单线程相比);


对于限定单个CPU的环境来说,Serial收集器没有线程交互(切换)开销,可以获得最高的单线程收集效率;


在用户的桌面应用场景中,可用内存一般不大(几十M至一两百M),可以在较短时间内完成垃圾收集(几十MS至一百多MS),只要不频繁发生,这是可以接受的


③、设置参数


“-XX:+UseSerialGC”:添加该参数来显式的使用串行垃圾收集器;


④Stop TheWorld说明


JVM在后台自动发起和自动完成的,在用户不可见的情况下,把用户正常的工作线程全部停掉,即GC停顿;


会带给用户不良的体验;


从JDK1.3到现在,从Serial收集器-》Parallel收集器-》CMS-》G1,用户线程停顿时间不断缩短,但仍然无法完全消除;


单线程地好处就是减少上下文切换,减少系统资源的开销。但这种方式的缺点也很明显,在GC的过程中,会暂停程序的执行。若GC不是频繁发生,这或许是一个不错的选择,否则将会影响程序的执行性能。 对于新生代来说,区域比较小,停顿时间短,所以比较使用。


2.ParNew收集器



ParNew收集器是Serial收集器新生代的多线程实现,注意在进行垃圾回收的时候依然会stop the world,只是相比较Serial收集器而言它会运行多条进程进行垃圾回收。


ParNew收集器在单CPU的环境中绝对不会有比Serial收集器更好的效果,甚至由于存在线程交互的开销,该收集器在通过超线程技术实现的两个CPU的环境中都不能百分之百的保证能超越Serial收集器。当然,随着可以使用的CPU的数量增加,它对于GC时系统资源的利用还是很有好处的。它默认开启的收集线程数与CPU的数量相同,在CPU非常多(譬如32个,现在CPU动辄4核加超线程,服务器超过32个逻辑CPU的情况越来越多了)的环境下,可以使用-XX:ParallelGCThreads参数来限制垃圾收集的线程数。


-UseParNewGC: 打开此开关后,使用ParNew + Serial Old的收集器组合进行内存回收,这样新生代使用并行收集器,老年代使用串行收集器。



ParNew垃圾收集器是Serial收集器的多线程版本。


①特点


除了多线程外,其余的行为、特点和Serial收集器一样;


如Serial收集器可用控制参数、收集算法、Stop The World、内存分配规则、回收策略等;


两个收集器共用了不少代码;


ParNew/Serial Old组合收集器运行示意图如下:


5d4c6812c8535adbb050f4ddf2e1bce8.png


②应用场景


在Server模式下,ParNew收集器是一个非常重要的收集器,因为除Serial外,目前只有它能与CMS收集器配合工作;


但在单个CPU环境中,不会比Serail收集器有更好的效果,因为存在线程交互开销。


③设置参数


“-XX:+UseConcMarkSweepGC”:指定使用CMS后,会默认使用ParNew作为新生代收集器;


“-XX:+UseParNewGC”:强制指定使用ParNew;


“-XX:ParallelGCThreads”:指定垃圾收集的线程数量,ParNew默认开启的收集线程与CPU的数量相同;


④为什么只有ParNew能与CMS收集器配合


CMS是HotSpot在JDK1.5推出的第一款真正意义上的并发(Concurrent)收集器,第一次实现了让垃圾收集线程与用户线程(基本上)同时工作;


CMS作为老年代收集器,但却无法与JDK1.4已经存在的新生代收集器Parallel Scavenge配合工作;


因为Parallel Scavenge(以及G1)都没有使用传统的GC收集器代码框架,而另外独立实现;而其余几种收集器则共用了部分的框架代码;


3.Parallel Scavenge收集器


46a9d80a6e05e4e3b19d57a0ee70bcdf.png


Parallel是采用复制算法的多线程新生代垃圾回收器,似乎和ParNew收集器有很多的相似的地方。但是Parallel Scanvenge收集器的一个特点是它所关注的目标是吞吐量(Throughput)。所谓吞吐量就是CPU用于运行用户代码的时间与CPU总消耗时间的比值,即吞吐量=运行用户代码时间 / (运行用户代码时间 + 垃圾收集时间)。停顿时间越短就越适合需要与用户交互的程序,良好的响应速度能够提升用户的体验;而高吞吐量则可以最高效率地利用CPU时间,尽快地完成程序的运算任务,主要适合在后台运算而不需要太多交互的任务。


Parallel Old收集器是Parallel Scavenge收集器的老年代版本,采用多线程和”标记-整理”算法。这个收集器是在jdk1.6中才开始提供的,在此之前,新生代的Parallel Scavenge收集器一直处于比较尴尬的状态。原因是如果新生代Parallel Scavenge收集器,那么老年代除了Serial Old(PS MarkSweep)收集器外别无选择。由于单线程的老年代Serial Old收集器在服务端应用性能上的”拖累“,即使使用了Parallel Scavenge收集器也未必能在整体应用上获得吞吐量最大化的效果,又因为老年代收集中无法充分利用服务器多CPU的处理能力,在老年代很大而且硬件比较高级的环境中,这种组合的吞吐量甚至还不一定有ParNew加CMS的组合”给力“。直到Parallel Old收集器出现后,”吞吐量优先“收集器终于有了比较名副其实的应用,在注重吞吐量及CPU资源敏感的场合,都可以优先考虑Parallel Scavenge加Parallel Old收集器。


-UseParallelGC: 虚拟机运行在Server模式下的默认值,打开此开关后,使用Parallel Scavenge + Serial Old的收集器组合进行内存回收。-UseParallelOldGC: 打开此开关后,使用Parallel Scavenge + Parallel Old的收集器组合进行垃圾回收


Parallel Scavenge垃圾收集器因为与吞吐量关系密切,也称为吞吐量收集器(Throughput Collector)。


①特点


(A)、有一些特点与ParNew收集器相似


新生代收集器;


采用复制算法;


多线程收集;


(B)、主要特点是:它的关注点与其他收集器不同


CMS等收集器的关注点是尽可能地缩短垃圾收集时用户线程的停顿时间;


而Parallel Scavenge收集器的目标则是达一个可控制的吞吐量(Throughput);


关于吞吐量与收集器关注点说明详见本节后面;


②应用场景


高吞吐量为目标,即减少垃圾收集时间,让用户代码获得更长的运行时间;


当应用程序运行在具有多个CPU上,对暂停时间没有特别高的要求时,即程序主要在后台进行计算,而不需要与用户进行太多交互;


例如,那些执行批量处理、订单处理、工资支付、科学计算的应用程序;


③设置参数


Parallel Scavenge收集器提供两个参数用于精确控制吞吐量:


(A)、“-XX:MaxGCPauseMillis”


控制最大垃圾收集停顿时间,大于0的毫秒数;


MaxGCPauseMillis设置得稍小,停顿时间可能会缩短,但也可能会使得吞吐量下降;


因为可能导致垃圾收集发生得更频繁;


(B)、“-XX:GCTimeRatio”


设置垃圾收集时间占总时间的比率,0<n<100的整数;


GCTimeRatio相当于设置吞吐量大小;


垃圾收集执行时间占应用程序执行时间的比例的计算方法是:


1 / (1 + n)


例如,选项-XX:GCTimeRatio=19,设置了垃圾收集时间占总时间的5%–1/(1+19);


默认值是1%–1/(1+99),即n=99;


垃圾收集所花费的时间是年轻一代和老年代收集的总时间;


如果没有满足吞吐量目标,则增加代的内存大小以尽量增加用户程序运行的时间;


此外,还有一个值得关注的参数:


(C)、“-XX:+UseAdptiveSizePolicy”


开启这个参数后,就不用手工指定一些细节参数,如:


新生代的大小(-Xmn)、Eden与Survivor区的比例(-XX:SurvivorRation)、晋升老年代的对象年龄(-XX:PretenureSizeThreshold)等;


JVM会根据当前系统运行情况收集性能监控信息,动态调整这些参数,以提供最合适的停顿时间或最大的吞吐量,这种调节方式称为GC自适应的调节策略(GC Ergonomiscs);


这是一种值得推荐的方式:


(1)、只需设置好内存数据大小(如"-Xmx"设置最大堆);


(2)、然后使用"-XX:MaxGCPauseMillis"或"-XX:GCTimeRatio"给JVM设置一个优化目标;


(3)、那些具体细节参数的调节就由JVM自适应完成;


这也是Parallel Scavenge收集器与ParNew收集器一个重要区别;


更多目标调优和GC自适应的调节策略说明请参考:


《Memory Management in the Java HotSpot™ Virtual Machine》 5节 Ergonomics – Automatic Selections and Behavior Tuning:http://www.oracle.com/technetwork/java/javase/tech/memorymanagement-whitepaper-1-150020.pdf


《Java Platform, Standard Edition HotSpot Virtual Machine Garbage Collection Tuning Guide》 第2节 Ergonomics:http://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/ergonomics.html#ergonomics


④吞吐量与收集器关注点说明


A)、吞吐量(Throughput)

CPU用于运行用户代码的时间与CPU总消耗时间的比值;


即吞吐量=运行用户代码时间/(运行用户代码时间+垃圾收集时间);


高吞吐量即减少垃圾收集时间,让用户代码获得更长的运行时间;


B)、垃圾收集器期望的目标(关注点)

1)、停顿时间

停顿时间越短就适合需要与用户交互的程序;


良好的响应速度能提升用户体验;


2)、吞吐量

高吞吐量则可以高效率地利用CPU时间,尽快完成运算的任务;


主要适合在后台计算而不需要太多交互的任务;


3)、覆盖区(Footprint)

在达到前面两个目标的情况下,尽量减少堆的内存空间;


可以获得更好的空间局部性;



相关文章
|
8月前
|
存储 算法 Java
深入理解JVM - ZGC收集器
深入理解JVM - ZGC收集器
225 1
|
6月前
|
存储 算法 安全
(八)JVM成神路之GC分区篇:G1、ZGC、ShenandoahGC高性能收集器深入剖析
在《GC分代篇》中,我们曾对JVM中的分代GC收集器进行了全面阐述,而在本章中重点则是对JDK后续新版本中研发推出的高性能收集器进行深入剖析。
213 12
|
6月前
|
算法 安全 Java
(七)JVM成神路之GC分代篇:分代GC器、CMS收集器及YoungGC、FullGC日志剖析
在《GC基础篇》中曾谈到过分代以及分区回收的概念,但基础篇更多的是建立在GC的一些算法理论上进行高谈阔论,而本篇则重点会对于分代收集器的实现进行全面详解,其中会涵盖串行收集器、并行收集器、三色标记、SATB算法、GC执行过程、并发标记、CMS收集器等知识,本篇则偏重于分析GC机制的落地实现,也就是垃圾收集器(Garbage Collector)。
167 8
|
存储 算法 Java
深入理解JVM - G1收集器
​ 上一篇通过案例说明了老年代的常见优化和处理方式,这一节来看下目前最为热门的G1收集器,G1收集器也是JDK9服务端默认的垃圾收集器,虽然JDK9在现在看来还不是十分的普及,但是学习这个垃圾收集器是十分重要也是十分必要的。
143 0
深入理解JVM - G1收集器
|
存储 算法 Oracle
深入理解JVM - ZGC收集器
上文讲到了Shenadoah收集器,这一节我们来讲一下ZGC收集器,ZGC收集器是JDK11之后由Oracle官方开发的一款低延迟垃圾收集器。另外这里吐槽一句ZGC的内容非常复杂并且知识点巨多,所以建议泡杯茶边喝边看。
232 0
|
算法 NoSQL Java
深入理解JVM - CMS收集器
上一篇文章我们讲解分代的基础理论,同时讲解了新生代和老年代各自的算法复制算法和标记整理算法,之后我们总结了新生代进入老年代的条件,在最后我们介绍的引用类型,同时进行了练习的提问和相关的解答。
100 0
|
算法 Java
JVM--JVM经典垃圾收集器整理(Serial收集器、ParNew收集器、Parallel Scavenge收集器、Garbage First收集器、Z
- 1、Serial收集器 - 2、ParNew收集器 - 3、Parallel Scavenge收集器 - 4、Serial Old收集器 - 5、Parallel Old收集器
395 0
JVM--JVM经典垃圾收集器整理(Serial收集器、ParNew收集器、Parallel Scavenge收集器、Garbage First收集器、Z
|
2月前
|
缓存 Prometheus 监控
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
434 1
|
16天前
|
存储 Java 程序员
【JVM】——JVM运行机制、类加载机制、内存划分
JVM运行机制,堆栈,程序计数器,元数据区,JVM加载机制,双亲委派模型
|
1月前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。