核心思想

标记-整理算法同样分为两个主要阶段,但第二个阶段有所不同:

  1. 标记阶段: 与标记-清除算法完全一致。遍历所有可达对象(从 GC Roots 开始),标记它们为“存活”。

  2. 整理阶段: 不再简单地清除垃圾对象,而是将所有存活的对象向内存空间的一端(通常是起始地址或结束地址)移动紧凑排列。移动完成后,边界之外的内存空间全部被视为空闲空间,可以一次性分配。

算法步骤详解

  1. 暂停应用程序线程:

    • 同样需要 “Stop-The-World” 停顿,以确保对象引用关系在标记和移动过程中保持稳定。标记-整理算法的 STW 时间通常比标记-清除更长,因为它包含了对象的移动开销。

  2. 标记阶段:

    • 起点: 从 GC Roots 开始。

    • 遍历: 采用深度优先搜索或广度优先搜索遍历对象图。

    • 标记: 将所有从 GC Roots 可达的对象标记为“存活”。结果与标记-清除算法相同。

  3. 整理阶段: 这是算法的核心改进点。

    • 计算新地址:

      • 遍历堆内存(通常是线性扫描)。

      • 计算每个存活对象在整理后应该被移动到的新地址。新地址通常是连续的,从堆空间的某一端(如低地址端)开始排列。

      • 一种常见的策略是:维护一个“指针”(compaction pointer),初始指向目标区域(如堆起始地址)。每遇到一个存活对象,就计算其大小,将该对象的新地址设置为当前指针位置,然后将指针向后移动该对象大小的距离。

    • 更新引用:

      • 由于对象被移动了位置,所有指向这些移动对象的引用(包括 GC Roots 中的引用和存活对象内部字段的引用)都需要被更新为对象的新地址。

      • 这一步需要再次遍历对象图(从 GC Roots 开始),访问所有存活对象及其引用,将指向被移动对象的引用值修改为计算好的新地址。

      • 关键点: 更新引用必须在对象实际移动之前完成,或者在移动过程中使用特殊的技巧(如 Brooks 指针)来保证引用的正确性。否则,移动对象后,旧的引用就会指向无效地址。

    • 移动对象:

      • 遍历堆内存(通常是线性扫描)。

      • 将每个存活对象复制(移动)到其在步骤 1 中计算好的新地址。

      • 移动完成后,所有存活对象都被紧密地排列在堆空间的一端(如低地址端)。

    • 回收空间:

      • 移动完成后,从最后一个存活对象之后的位置开始,直到堆空间的另一端(如高地址端),所有内存空间都是连续的空闲空间

      • 这个连续的大块空闲内存可以被一个简单的指针(如 bump pointer)管理,新对象的分配变得极其高效(只需移动指针并清零内存)。

关键特点与优缺点

  • 优点:

    • 解决内存碎片: 这是最核心的优势!通过将存活对象紧凑排列,消除了标记-清除算法产生的内存碎片问题。分配新对象时,只需要在连续空闲空间的末尾进行指针碰撞(bump-the-pointer),分配速度非常快且简单。

    • 空间利用率高: 消除了碎片浪费,堆空间得到更有效的利用。特别是对于需要分配大对象的场景,成功率更高。

    • 局部性原理提升: 紧凑排列的对象在内存中位置相邻,更有可能被加载到 CPU 缓存同一缓存行(Cache Line)中。这可以提升程序访问对象的效率(缓存命中率提高)。

  • 缺点:

    • STW 时间更长: 移动对象和更新引用的开销通常比简单的清除操作要大得多。这导致 Stop-The-World 停顿时间通常比标记-清除算法更长,尤其是在堆很大、存活对象很多的情况下。这是标记-整理算法最主要的缺点。

    • 移动开销: 复制对象本身需要时间,尤其是对于大对象。

    • 更新引用开销: 需要遍历整个对象图来更新所有指向被移动对象的引用,这也是一个耗时的操作。

    • 实现更复杂: 需要精确地处理对象移动和引用更新,实现难度高于标记-清除算法。

应用场景与演变

  • 典型应用:

    • 老年代回收: 标记-整理算法因其能有效解决碎片问题,非常适合老年代的垃圾回收。老年代对象的特点是:

      • 存活率高(每次GC后大部分对象依然存活)。

      • 对象存活时间长。

      • 分配频率相对新生代较低。

      • 内存碎片非常敏感(长期运行后碎片累积会导致 Full GC 失败)。

    • Serial Old: HotSpot JVM 中的老年代串行收集器就使用标记-整理算法。

    • Parallel Old: HotSpot JVM 中的老年代并行收集器也使用标记-整理算法(多线程并行执行标记和整理)。

    • CMS 的备胎: 当 CMS(Concurrent Mark-Sweep)收集器发生 Concurrent Mode Failure(并发收集跟不上对象分配速度)或 晋升失败(Promotion Failed),或者堆中碎片过多时,CMS 会退回到 Serial Old 收集器(标记-整理)进行一次 Full GC 来整理老年代碎片。

  • 现代收集器中的优化:

    • G1: 虽然 G1 的目标是可控停顿时间,并且主要使用复制算法在 Region 间转移存活对象,但其在全局层面也可以看作是一种更精细、更智能的标记-整理(通过选择性地回收和整理 Region 来减少碎片)。

    • ZGC / Shenandoah: 这些超低停顿收集器使用极其复杂的并发算法,但它们在回收阶段的核心目标之一也是移动对象进行整理(并发地或部分并发地),以消除碎片。它们通过读屏障等技术来实现并发移动对象和更新引用,极大地减少了 STW 时间(通常缩短到几毫秒级别),克服了传统标记-整理算法 STW 长的最大缺点。

与标记-清除算法的对比总结

特性标记-清除算法标记-整理算法
核心阶段标记 -> 清除标记 -> 整理(计算地址->更新引用->移动对象)
内存碎片严重,产生大量不连续碎片,产生连续大块空闲空间
分配速度慢(需搜索空闲列表)极快(指针碰撞)
STW 时间相对较短(主要耗时在标记)相对较长(标记 + 移动 + 更新引用)
移动对象
空间开销低(只需标记位)低(标记位+可能的额外空间记录新地址)
时间开销与存活对象数+堆大小成正比存活对象数成正比
局部性差(对象分散)(对象紧凑)
典型场景较少单独使用(如 CMS 的并发清除)老年代(Serial Old, Parallel Old)

总结

标记-整理算法通过引入对象移动和紧凑排列的整理阶段,完美解决了标记-清除算法最致命的内存碎片问题,带来了更高的内存利用率更快的对象分配速度(指针碰撞)。然而,这种优势是以更长的 Stop-The-World 停顿时间(主要来自移动对象和更新引用)为代价的。

因此,它非常适合对碎片敏感但能容忍较长停顿的老年代垃圾回收。传统的 Serial Old 和 Parallel Old 收集器就是其代表。现代超低停顿收集器(如 ZGC, Shenandoah)通过并发移动和读屏障等革命性技术,极大地克服了 STW 长的缺点,将标记-整理的思想推向了新的高度,使其能够应用于对延迟极其敏感的系统中。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/web/84193.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/84193.shtml
英文地址,请注明出处:http://en.pswp.cn/web/84193.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

进程虚拟地址空间

1. 程序地址空间回顾 我们在学习语言层面时,会了解到这样的空间布局图,我们先对他进行分区了解: 如果以静态static修饰的变量就会当成已初始化全局变量来看待,存放在已初始化数据区和未初始化数据区之前。 如果不用static修饰test…

C语言学习day17-----位运算

目录 1.位运算 1.1基础知识 1.1.1定义 1.1.2用途 1.1.3软件控制硬件 1.2运算符 1.2.1与 & 1.2.2或 | 1.2.3非 ~ 1.2.4异或 ^ 1.2.5左移 << 1.2.6右移 >> 1.2.7代码实现 1.2.8置0 1.2.9置1 1.2.10不借助第三方变量&#xff0c;实现两个数的交换…

【linux】简单的shell脚本练习

简单易学 解释性语言&#xff0c;不需要编译即可执行 对于一个合格的系统管理员来说&#xff0c;学习和掌握Shell编程是非常重要的&#xff0c;通过shell程序&#xff0c;可以在很大程度上简化日常的维护工作&#xff0c;使得管理员从简单的重复劳动中解脱出来 用户输入任意两…

机构运动分析系统开发(Python实现)

机构运动分析系统开发(Python实现) 一、引言 机构运动分析是机械工程的核心内容,涉及位置、速度和加速度分析。本系统基于Python开发,实现了平面连杆机构的完整运动学分析,包含数学建模、数值计算和可视化功能。 二、系统架构设计 #mermaid-svg-bT8TPKQ98UU9ERet {font…

工程师生活:清除电热水壶(锅)水垢方法

清除电热水壶&#xff08;锅&#xff09;水垢方法 水垢是水加热时自然形成的钙质沉淀物&#xff0c;常粘附在水壶内壁及发热盘上。它不仅影响水的品质&#xff0c;还会缩短水壶的使用寿命&#xff0c;因此需要定期清除。建议根据各地水质不同&#xff0c;每年除垢 2 至 4 次。…

[分布式并行策略] 数据并行 DP/DDP/FSDP/ZeRO

上篇文章【[论文品鉴] DeepSeek V3 最新论文 之 DeepEP】 介绍了分布式并行策略中的EP&#xff0c;简单的提到了其他几种并行策略&#xff0c;但碍于精力和篇幅限制决定将内容分几期&#xff0c;本期首先介绍DP&#xff0c;但并不是因为DP简单&#xff0c;相反DP的水也很深&…

LeeCode144二叉树的前序遍历

项目场景&#xff1a; 给你二叉树的根节点 root &#xff0c;返回它节点值的 前序 遍历。 示例 1&#xff1a; 输入&#xff1a;root [1,null,2,3] 输出&#xff1a;[1,2,3] 解释&#xff1a; 示例 2&#xff1a; 输入&#xff1a;root [1,2,3,4,5,null,8,null,null,6,7…

日本生活:日语语言学校-日语作文-沟通无国界(3)-题目:わたしの友達

日本生活&#xff1a;日语语言学校-日语作文-沟通无国界&#xff08;&#xff13;&#xff09;-题目&#xff1a;わたしの友達 1-前言2-作文原稿3-作文日语和译本&#xff08;1&#xff09;日文原文&#xff08;2&#xff09;对应中文&#xff08;3&#xff09;对应英文 4-老师…

使用 rsync 拉取文件(从远程服务器同步到本地)

最近在做服务器迁移&#xff0c;文件好几个T。。。。只能单向访问&#xff0c;服务器。怎么办&#xff01;&#xff01;&#xff01; 之前一直是使用rsync 服务器和服务器之间的双向同步、备份&#xff08;这是推的&#xff09;。现在服务器要迁移&#xff0c;只能单向访问&am…

Linux 并发编程:从线程池到单例模式的深度实践

文章目录 一、普通线程池&#xff1a;高效线程管理的核心方案1. 线程池概念&#xff1a;为什么需要 "线程工厂"&#xff1f;2. 线程池的实现&#xff1a;从 0 到 1 构建基础框架 二、模式封装&#xff1a;跨语言线程库实现1. C 模板化实现&#xff1a;类型安全的泛型…

2013年SEVC SCI2区,自适应变领域搜索算法Adaptive VNS+多目标设施布局,深度解析+性能实测

目录 1.摘要2.自适应局部搜索原理3.自适应变领域搜索算法Adaptive VNS4.结果展示5.参考文献6.代码获取7.算法辅导应用定制读者交流 1.摘要 VNS是一种探索性的局部搜索方法&#xff0c;其基本思想是在局部搜索过程中系统性地更换邻域。传统局部搜索应用于进化算法每一代的解上&…

详细介绍医学影像显示中窗位和窗宽

在医学影像&#xff08;如DICOM格式的CT图像&#xff09;中&#xff0c;**窗宽&#xff08;Window Width, WW&#xff09;和窗位&#xff08;Window Level, WL&#xff09;**是两个核心参数&#xff0c;用于调整图像的显示对比度和亮度&#xff0c;从而优化不同组织的可视化效果…

Unity_VR_如何用键鼠模拟VR输入

文章目录 [TOC] 一、创建项目1.直接创建VR核心模板&#xff08;简单&#xff09;2.创建3D核心模板导入XR包 二、添加XR设备模拟器1.打开包管理器2.添加XR设备模拟器3.将XR设备模拟器拖到场景中4.运行即可用键盘模拟VR输入 一、创建项目 1.直接创建VR核心模板&#xff08;简单&…

SpringBoot定时监控数据库状态

1.application.properties配置文件 # config for mysql spring.datasource.url jdbc\:mysql\://127.0.0.1\:3306/数据库名?characterEncoding\utf8&useSSL\false spring.datasource.username 账号 spring.datasource.password 密码 spring.datasource.validation-quer…

Qt联合Halcon开发一:Qt配置Halcon环境【详细图解流程】

在Qt中使用Halcon库进行图像处理开发&#xff0c;可以有效地结合Qt的图形界面和Halcon强大的计算机视觉功能。下面是详细的配置过程&#xff0c;帮助你在Qt项目中成功集成Halcon库。 步骤 1: 安装Halcon软件并授权 首先&#xff0c;确保你已经在电脑上安装了Halcon软件&#x…

一体化(HIS系统)医院信息系统,让医疗数据互联互通

在医疗信息化浪潮下&#xff0c;HIS系统、LIS系统、PACS系统、电子病历系统等信息系统成为医疗机构必不可少的一部分&#xff0c;从患者挂号到看诊&#xff0c;从各种检查到用药&#xff0c;从院内治疗到院外管理……医疗机构不同部门、不同科室的各类医疗、管理业务几乎都初步…

Spring Boot 的 3 种二级缓存落地方式

在高并发系统设计中&#xff0c;缓存是提升性能的关键策略之一。随着业务的发展&#xff0c;单一的缓存方案往往无法同时兼顾性能、可靠性和一致性等多方面需求。 此时&#xff0c;二级缓存架构应运而生&#xff0c;本文将介绍在Spring Boot中实现二级缓存的三种方案。 一、二…

Android Studio Profiler使用

一:memory 参考文献: AndroidStudio之内层泄漏工具Profiler使用指南_android studio profiler-CSDN博客

Zephyr boot

<!DOCTYPE html> <html lang"zh-CN"> <head> <meta charset"UTF-8"> <meta name"viewport" content"widthdevice-width, initial-scale1.0"> <title>Zephyr设备初始化机制交互式解析…

腾讯地图Web版解决热力图被轮廓覆盖的问题

前言 你好&#xff0c;我是喵喵侠。 还记得那天傍晚&#xff0c;我正对着电脑调试一个腾讯地图的热力图页面。项目是一个区域人流密度可视化模块&#xff0c;我加了一个淡蓝色的轮廓图层用于表示区域范围&#xff0c;热力图放在下面用于展示人流热度。效果一预览&#xff0c;…