1 GC分类与性能指标
1.1 垃圾回收器概述
-
垃圾收集器
没有在规范中进行过多的规定,可以由不同的厂商、不同版本的JVM来实现。
- 由于JDK的版本处于高速迭代过程中,因此Java发展至今已经衍生了众多的Gc版本。
- 从不同角度分析垃圾收集器,可以将GC分为不同的类型。
1.2 垃圾回收器分类
1.2.1 按照线程分
按照线程分,可以分为串行垃圾回收器和并行垃圾回收器
-
串行回收指的是在同一时间段内只允许有一个CPU用于执行垃圾回收操作,此时工作线程被暂停,直至垃圾收集工作结束。
①在诸如单CPU处理器或者较小的应用内存等硬件平台不是特别优越的
场合,串行回收器的性能表现可以超过并行回收器和并发回收器。所以,串行回收默认被应用在客户端的client模式下的JVM中
② 在并发能力比较强的CPU上,并行回收器产生的停顿时间要短于串行回
收器。
- 和串行回收相反,并行收集可以运用多个CPU同时执行垃圾回收,因此提升了应用的吞吐量,不过并行回收仍然与串行回收一样,采用独占式,使用了“stop-the-world”机制。
1.2.2 按照工作模式分
按照功能模式分,可以分为并发式垃圾回收器和独占式垃圾回收器
-
并发式垃圾回收器
与应用程序线程交替工作,以尽可能减少应用程序的停顿时间。
-
独占式垃圾回收器
(stop-the-world)一旦运行,就停止应用程序中的所有用户线程,直到垃圾回收过程完全结束。
1.2.3 碎片处理方式分
按照碎片处理方式分,可以分为压缩式垃圾回收器和非压缩式垃圾回收器
- 压缩式垃圾回收器会在回收完成后,对存活对象进行压缩整理,消除回收后的碎片(再分配对象空间使用:指针碰撞)。
- 非压缩式的垃圾回收器不进行这步操作(再分配对象空间使用:空闲列表)。
-
1.2.4 工作的内存区间分
按照工作的内存区间分,可以分为年轻代垃圾回收器和老年代垃圾回收器
1.3 评估GC的性能指标
-
吞吐量:运行用户代码的时间占总运行时间的比例(总运行时间:程序的运行时间+内存回收的时间)。
- 垃圾收集开销:吞吐量的补数,垃圾收集所用时间与总运行时间的比例。
- 暂停时间:执行垃圾收集时,程序的工作线程被暂停的时间。
- 收集频率:相对于应用程序的执行,收集操作发生的频率。
- 内存占用: Java堆区所占的内存大小。
- 快速:一个对象从诞生到被回收所经历的时间。
在所有的性能指标中标红的三个指标是最重要的。
①这三者共同构成一个“不可能三角”。三者总体的表现会随着技术进步而越来越好。一款优秀的收集器通常最多同时满足其中的两项。
②这三项里,暂停时间的重要性日益凸显。因为随着硬件发展,内存占用多些越来越能容忍,硬件性能的提升也有助于降低收集器运行时对应用程序的影响,即提高了吞吐量。而内存的扩大,对延迟反而带来负面效果。
③简单来说,主要抓住两点:吞吐量 和 暂停时间
1.4 吞吐量与暂停时间的对比说明
1.4.1 吞吐量
-
吞吐量
就是CPU用于运行用户代码的时间与CPU总消耗时间的比值,即吞吐量=运行用户代码时间 / (运行用户代码时间+垃圾收集时间)。
比如:虚拟机总共运行了100分钟,其中垃圾收集花掉1分钟,那吞吐量就是99%。
-
这种情况下,应用程序能容忍较高的暂停时间,因此,高吞吐量的应用程序有更长的时间基准,快速响应是不必考虑的。
-
吞吐量优先,意味着在单位时间内,STW的时间最短:0.2 + 0.2 = 0.4
1.4.2 暂停时间
-
“暂停时间”
是指一个时间段内应用程序线程暂停,让GC线程执行的状态
例如,GC期间100毫秒的暂停时间意味着在这100毫秒期间内没有应用
程序线程是活动的。
- 暂停时间优先,意味着尽可能让STW的时间最短:0.1 + 0.1 + 0.1 + 0.1 + 0.1 = 0.5
1.4.3 高吞吐量 VS 暂停时间
- 高吞吐量较好因为这会让应用程序的最终用户感觉只有应用程序线程在做“生产性”工作。直觉上,吞吐量越高程序运行越快。
- 低暂停时间(低延迟)较好因为从最终用户的角度来看不管是GC还是其他原因导致一个应用被挂起始终是不好的。这取决于应用程序的类型,有时候甚至短暂的200毫秒暂停都可能打断终端用户体验。因此,具有低的较大暂停时间是非常重要的,特别是对于一个交互式应用程序。
-
不幸的是”高吞吐量”和”低暂停时间”是一对相互竞争的目标(矛盾)。
①因为如果选择以吞吐量优先,那么必然需要降低内存回收的执行频率,但是这样
会导致GC需要更长的暂停时间来执行内存回收。
②相反的,如果选择以低延迟优先为原则,那么为了降低每次执行内存回收时的暂停时间,也只能频繁地执行内存回收,但这又引起了年轻代内存的缩减和导致程序吞吐量的下降。
在设计(或使用)GC算法时,我们必须确定我们的目标:一个GC算法只可能针对两个目标之一(即只专注于较大吞吐量或最小暂停时间)或尝试找到一个二者的折衷。
现在标准: 在最大吞吐量优先的情况下,降低停顿时间。
2 垃圾回收器的迭代发展史
2.1 不同垃圾回收器的概述
垃圾收集机制是Java的招牌能力,极大地提高了开发效率。那么,Java常见的垃圾收集器有哪些?
有了虚拟机,就一定需要收集垃圾的机制,这就是Garbage Collection,对应的产品我们称为Garbage Collector。
- 1999年随JDK1.3.1一起来的是串行方式的serial GC,它是第一款GC。ParNew垃圾收集器是serial收集器的多线程版本。
- 2002年2月26日,Parallel GC和Concurrent Mark Sweep Gc跟随JDK1.4.2一起发布。
- Parallel GC在JDK6之后成为HotSpot默认GC。2012年,在JDK1.7u4版本中,G1可用。
- 2017年,JDK9中G1变成默认的垃圾收集器,以替代CMS。
- 2018年3月,JDK10中G1垃圾回收器的并行完整垃圾回收,实现并行性来改善最坏情况下的延迟。
- 2018年9月,JDK11发布。引入Epsilon垃圾回收器,又被称为"No-Op(无操作)“回收器。同时,引入ZGC:可伸缩的低延迟垃圾回收(Experimental)。
- 2019年3月,JDK12发布。增强G1,自动返回未用堆内存给操作系统。同时,引入shenandoah Gc:低停顿时间的GC (Experimental)。
- 2019年9月,JDK13发布。增强zGC,自动返回未用堆内存给操作系统。
- 2020年3月,JDK14发布。删除CMs垃圾回收器。扩展ZGC在macos和windows上的应用。
2.2 七款经典的垃圾回收器
- 串行回收器:Serial、Serial Old
- 并行回收器: ParNew、Parallel scavenge、Parallel old
-
并发回收器:CMS、G1
2.3 七款经典的垃圾回收器与垃圾分代之间的关系
- 新生代收集器:Serial、ParNew、Parallel scavenge
- 老年代收集器: Serial old、Parallel old、CMS
- 整堆收集器:G1
2.4 垃圾回收器的组合关系
为什么要有很多收集器,一个不够吗? 因为Java的使用场景很多,移动端,服务器等。所以就需要针对不同的场景,提供不同的垃圾收集器,提高垃圾收集的性能。
虽然我们会对各个收集器进行比较,但并非为了挑选一个最好的收集器出来。没有一种放之四海皆准、任何场景下都适用的完美收集器存在,更加没有万能的收集器。所以我们选择的只是对具体应用最合适的收集器。
2.5 查看默认的垃圾回收器
2.5.1 准备工作
编写代码
import java.util.ArrayList;
public class GCUseTest {
public static void main(String[] args) {
ArrayList<byte[]> list = new ArrayList<>();
while (true) {
byte[] arr = new byte[100];
list.add(arr);
try {
Thread.sleep(10);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}
}
}
配置JDK1.8环境
2.5.2 方式一
-XX:+:PrintCommandLineFlags:查看命令行相关参数(包含垃圾收集器)
运行代码
通过运行结果发现使用的是ParallelGC
2.5.3 方式二
使用命令行指令:jinfo -flag 相关垃圾回收器参数 进程ID
①运行程序
②在命令行窗口查看Java进程
GCUseTest进程ID为14168
③查看垃圾回收器
如果结果显示+号则表示使用该垃圾回收器。
3 Serial回收器: 串行回收
3.1 Serial 回收器与Serial Old回收器介绍
-
Serial收集器
是最基本、历史最悠久的垃圾收集器了。JDK1.3之前回收新生代唯一的选择。
-
Serial收集器作为HotSpot中Client模式下的默认新生代垃圾收集器
-
Serial收集器采用复制算法、串行回收和“stop-the-world”机制的方式执行内存回收。
-
除了年轻代之外,Serial收集器还提供用于执行老年代垃圾收集的Serial Old收集器
。Serial Old收集器同样也采用了串行回收和“stop-the-world”机制,只不过内存回收算法使用的是标记-压缩算法。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)