JVM05-垃圾收集算法

简介: 上一篇我们介绍了JVM04-JVM中内存溢出以及其处理方法。这一篇文章我们来熟悉下JVM中各种垃圾回收算法。这些垃圾收集算法是后面各种垃圾收集器的算法基础。闲话少叙,让我们直入主题。

前言

上一篇我们介绍了JVM04-JVM中内存溢出以及其处理方法。这一篇文章我们来熟悉下JVM中各种垃圾回收算法。这些垃圾收集算法是后面各种垃圾收集器的算法基础。闲话少叙,让我们直入主题。

标记-清除算法

标记-清除算法分为"标记"和"清除"两个阶段,首先标记所有需要回收的对象,在标记完成后,统一回收掉所有被标记的对象,也可以反过来,标记存活的对象,统一回收未标记的对象,标记的过程就是对象是否属于垃圾的判定过程,一般是通过可达性分析算法,也就是说某个对象到GC Root间是否有引用链相连,如果没有则判断该对象不再被使用,也就是说是可以被回收的。如下图所示,标黄的内存块,在标记之后,就被清除了,留下来不少的内存碎片。

标记-清除算法的优点

1.实现简单

标记-清除算法实现简单, 与其他算法的组合也相应地简单。

2.与保守式GC算法兼容中,对象是不能被移动的,因此保守式GC算法跟把对象从现在的场所复制算法与标记-压缩算法不兼容。标记-清除算法因为不会移动对象,所以非常适合搭配保守式GC算法。事实上,在很多采用保守式GC算法的处理程序中也用到了标记-清除算法。

标记-清除算法的缺点

1.分配速度

执行效率不稳定,如果Java堆中包含大量对象,而且这些对象大部分是需要回收的,这是必须进行大量标记和清除动作,导致标记和清除两个过程的执行效率都随着对象数量增长而降低。

2.碎片化

内存空间的碎片问题、标记、清除之后会产生大量不连续的内存碎片,空间碎片太多会导致当以后在程序运行过程中需要分配较大对象时无法找到足够连续内存而不得不提前触发另一次垃圾收集动作。

标记-复制算法

标记-复制算法将可用的内存容量划分为大小相等的两块,每次只使用其中的一块,当这一块的内存用完了,就将还存活着的对象复制到另外一块上面,然后再把已使用的内存空间一次清理掉。如果内存中多数对象都是存活的,这种算法将会产生大量的内存间复制的开销,但对于多数对象都是可回收的情况下,算法需要复制的就是占少数的存活对象。而且每次都是针对整个半区进行内存回收,分配内存时就不用考虑有内存碎片的复杂情况了。如下图所示:回收之后将存活的对象全部移动到原来的保留区域。

标记-复制算法的优点

1.优秀的吞吐量

标记-清除算法消耗的吞吐量是搜索活动对象(标记阶段)所花费的时间和搜索整体堆(清除阶段)所花费的时间之和。

另一方面,因为标记-复制算法只搜索并复制活动对象,所以跟一般的标记-清除算法相比,它能在较短时间内完成GC,也就是说,其吞吐量优秀。

尤其是堆越大,差距越明显。

2.可实现高速分配

标记-复制算法不使用空闲链表,这是因为分块是一块连续的内存空间,比起标记-清除算法等使用空闲链表的分配,标记-复制算法明显快得多。

3.不会发生碎片化

存活对象被几种安排到保留区域,像这样把对象重新集中,放在堆的一端的行为就叫作压缩,在标记-复制算法中,每次运行GC时都会执行压缩。因此复制算法不会发生碎片化。

标记-复制算法的缺点

1.堆使用效率低下
标记-复制算法把堆二等分,通常只能利用其中的一半来安排对象,也就是说,只有一半的堆能被使用,相比其他能使用整个堆的GC算法而言,这是标记-复制算法的一个重大缺陷。

2.不兼容保守式GC算法
标记-复制算法因为必须要移动对象重写指针,所以跟保守式GC算法不相容。

Appel式回收

在1989年,Andrew Appel针对具备"朝生夕灭"特点的对象,提出了一种更优化的半区复制分代策略,称之为"Appel式回收"。Appel式回收的具体做法就是把新生代分为一块较大的Eden和Survivor中仍然存活的对象一次性复制到另一块Survivor空间上,然后直接清理掉Eden和已用过的那块Survivor空间,HotSpot虚拟机 默认Eden和Survivor的大小比例是8:1,也即每次新生代可用空间为整个新生代容量的90%(Eden的80%加上一个Survivor的10%),只有一个Survivor空间,即10%的新生代会被"浪费"掉。

标记-整理算法

标记-复制算法在对象存活率较高时就要进行较多的复制操作,效率将会降低。更关键的是,如果不想浪费50%的空间,就需要有额外的空间进行分配担保,以应对被使用的内存中所有对象都100%存活的极端情况。所以老年代一般不能直接选用这种算法。所以,针对老年代的垃圾收集,有标记-整理算法,首先还是标记所有需要回收的对象,然后让所有存活的对象都向内存一端移动,然后,直接清理掉边界以外的内存。

如果移动存活对象,尤其是在老年代这种每次回收都有大量对象存活区域,移动存活对象并更新所有引用这些对象的地方将会是一种极为负重的操作。而且这种对象移动操作必须全程暂停用户应用程序才能进行。移动对象则内存回收时会更复杂,不移动则内存分配时会更复杂,从垃圾收集的停顿时间来看,不移动对象停顿时间会更短,甚至不需要停顿,但是从整个程序的吞吐量来看,移动对象会更划算。关注吞吐量的Parallel Scavenge收集器是基于标记-整理算法的,而关注延迟的CMS收集器则是基于标记-清除算法的。标记-整理算法清理过程如下图所示:

标记-整理算法的优点

1.标记-整理算法会执行压缩,和其他算法相比而言,堆利用效率高。而且标记-整理算法不会出现标记-复制算法那样只能利用半个堆的情况。另外,由于有了压缩过程,不会产生碎片化。

标记-整理算法的缺点

1.压缩花费计算成本
标记-清除算法中,清除阶段也要搜索整个堆,不过搜索1次就够了,但标记-压缩算法要搜索3次,这样就要花费约3倍的时间,这是一个相当巨大的缺陷,特别是堆越大,所消耗的成本也就越大。

保守式GC算法

前面提到了保守式,简单的来说,保守式GC(Conservative GC)指的是"不能识别指针和非指针的GC"。

总结

本文简单的介绍了JVM中几个基本的垃圾回收算法,主要是标记-清除算法,标记-复制算法和标记-整理算法。每个算法都有各自的优缺点。一般而言新生代采用标记-清除算法和标记-复制算法居多,老年代会采用标记-整理算法。

相关文章
|
3月前
|
缓存 算法 Java
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
这篇文章详细介绍了Java虚拟机(JVM)中的垃圾回收机制,包括垃圾的定义、垃圾回收算法、堆内存的逻辑分区、对象的内存分配和回收过程,以及不同垃圾回收器的工作原理和参数设置。
110 4
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
|
2月前
|
算法 Java
JVM有哪些垃圾回收算法?
(1)标记清除算法: 标记不需要回收的对象,然后清除没有标记的对象,会造成许多内存碎片。 (2)复制算法: 将内存分为两块,只使用一块,进行垃圾回收时,先将存活的对象复制到另一块区域,然后清空之前的区域。用在新生代 (3)标记整理算法: 与标记清除算法类似,但是在标记之后,将存活对象向一端移动,然后清除边界外的垃圾对象。用在老年代
23 0
|
3月前
|
存储 监控 算法
JVM调优深度剖析:内存模型、垃圾收集、工具与实战
【10月更文挑战第9天】在Java开发领域,Java虚拟机(JVM)的性能调优是构建高性能、高并发系统不可或缺的一部分。作为一名资深架构师,深入理解JVM的内存模型、垃圾收集机制、调优工具及其实现原理,对于提升系统的整体性能和稳定性至关重要。本文将深入探讨这些内容,并提供针对单机几十万并发系统的JVM调优策略和Java代码示例。
68 2
|
3月前
|
算法 Java
JVM进阶调优系列(4)年轻代和老年代采用什么GC算法回收?
本文详细介绍了JVM中的GC算法,包括年轻代的复制算法和老年代的标记-整理算法。复制算法适用于年轻代,因其高效且能避免内存碎片;标记-整理算法则用于老年代,虽然效率较低,但能有效解决内存碎片问题。文章还解释了这两种算法的具体过程及其优缺点,并简要提及了其他GC算法。
 JVM进阶调优系列(4)年轻代和老年代采用什么GC算法回收?
|
3月前
|
存储 算法 Java
【JVM】垃圾释放方式:标记-清除、复制算法、标记-整理、分代回收
【JVM】垃圾释放方式:标记-清除、复制算法、标记-整理、分代回收
76 2
|
3月前
|
缓存 算法 Java
GC垃圾收集算法
这篇文章详细讨论了垃圾收集(GC)的几种算法,包括引用计数、可达性分析、标记-清除、标记-复制和标记-整理算法,并介绍了这些算法的优缺点和适用场景。
45 0
GC垃圾收集算法
|
5月前
|
存储 算法 Java
JVM自动内存管理之垃圾收集算法
文章概述了JVM内存管理和垃圾收集的基本概念,提供一个关于JVM内存管理和垃圾收集的基础理解框架。
JVM自动内存管理之垃圾收集算法
|
5月前
|
存储 算法 Java
JVM组成结构详解:类加载、运行时数据区、执行引擎与垃圾收集器的协同工作
【8月更文挑战第25天】Java虚拟机(JVM)是Java平台的核心,它使Java程序能在任何支持JVM的平台上运行。JVM包含复杂的结构,如类加载子系统、运行时数据区、执行引擎、本地库接口和垃圾收集器。例如,当运行含有第三方库的程序时,类加载子系统会加载必要的.class文件;运行时数据区管理程序数据,如对象实例存储在堆中;执行引擎执行字节码;本地库接口允许Java调用本地应用程序;垃圾收集器则负责清理不再使用的对象,防止内存泄漏。这些组件协同工作,确保了Java程序的高效运行。
37 3
|
6月前
|
存储 监控 算法
(六)JVM成神路之GC基础篇:对象存活判定算法、GC算法、STW、GC种类详解
经过前面五个章节的分析后,对于JVM的大部分子系统都已阐述完毕,在本文中则开始对JVM的GC子系统进行全面阐述,GC机制也是JVM的重中之重,调优、监控、面试都逃不开的JVM话题。
190 8
|
5月前
|
C# UED 开发者
WPF打印功能实现秘籍:从页面到纸张,带你玩转WPF打印技术大揭秘!
【8月更文挑战第31天】在WPF应用开发中,打印功能至关重要,不仅能提升用户体验,还增强了应用的实用性。本文介绍WPF打印的基础概念与实现方法,涵盖页面元素打印、打印机设置及打印预览。通过具体案例,展示了如何利用`PrintDialog`和`PrintDocument`控件添加打印支持,并使用`PrinterSettings`类进行配置,最后通过`PrintPreviewWindow`实现打印预览功能。
534 0