G1 和 CMS

1、CMS

CMS(Concurrent Mark Sweep,并发标记清除,是为了解决早期垃圾收集器在执行垃圾回收时导致应用程序暂停时间过长的问题而设计的。

CMS的工作流程主要包括以下几个阶段:

  1. 初始标记(Initial Mark):这个阶段是Stop-The-World事件,所有应用线程暂停,垃圾收集器仅标记与根对象直接关联的对象。这个阶段很短。

  2. 并发标记(Concurrent Mark):在这个阶段,应用线程与垃圾收集线程并发执行。垃圾收集器会遍历整个堆,标记出所有存活的对象。这个过程比较耗时,但因为与用户线程并发执行,所以不会导致长时间的暂停。

  3. 重新标记(Remark):这是另一个短暂的Stop-The-World事件,用于修正并发标记期间因用户程序继续运行而可能产生的标记变动。相比初始标记,这个阶段稍长一些,因为它需要处理整个堆的信息。

  4. 并发清除(Concurrent Sweep):此阶段也是与用户线程并发执行的,垃圾收集器会清除那些被标记为死亡的对象所占用的内存空间,回收空间供后续使用。

CMS的优点在于其低延迟特性,能够在大部分垃圾回收工作与应用程序并发执行,从而减少因垃圾回收而导致的暂停时间。然而,CMS也有一些缺点,比如:

  • 对CPU资源消耗较高:由于其并发标记和清除的特性,CMS在运行时会占用较多的CPU资源。
  • 无法处理浮动垃圾(Floating Garbage):在并发标记和清除阶段,新产生的垃圾不能被这次收集处理,需要等到下一次收集。
  • 可能出现“Concurrent Mode Failure”:如果在老年代剩余空间不足以容纳新生代晋升的对象时,会导致CMS停止并发收集,转而执行完全的垃圾收集(Full GC),这会导致长时间的应用暂停。

尽管CMS在很长一段时间内被广泛使用于对延迟敏感的服务,但随着G1垃圾收集器的发展和完善,CMS在JDK 9中已被废弃,并在JDK 14中完全移除,推荐使用G1或ZGC、Shenandoah等现代垃圾收集器作为替代。

2、G1

G1垃圾收集器(Garbage First Garbage Collector)是Oracle在Java 7中引入的一种高性能垃圾收集器,旨在替代传统的CMS(Concurrent Mark-Sweep)垃圾收集器。它结合了多种垃圾收集技术,旨在提供低停顿、高吞吐量和高效的内存管理;

G1垃圾收集器的工作机制和主要算法:

分区(Region)机制

  • Region:G1将整个堆划分成多个大小相等的独立区域(Region),每个Region通常为1MB到32MB,根据堆的大小自动调整。
  • 灵活性:Region可以充当Eden区、Survivor区或老年代的一部分,这种设计使得内存管理更加灵活,有助于优化垃圾收集过程。

年轻代收集(Young GC)

  • 复制算法:新生代使用复制算法(Copying),将存活对象从Eden区和一个Survivor区复制到另一个Survivor区。这种方式可以快速整理内存,提高新生代回收的效率。

并发标记周期(Concurrent Marking Cycle)

  1. 初始标记(Initial Marking):暂停所有应用线程,标记从根对象(GC Roots)直接可达的对象。这个过程时间较短。
  2. 并发标记(Concurrent Marking):在应用线程运行时,进行可达性分析,标记所有可达的对象。这个过程中不需要暂停应用线程。
  3. 最终标记(Final Marking):稍微暂停应用线程以完成标记过程,包括处理并发标记阶段新创建的对象。
  4. 筛选回收(Selection Phase):根据之前标记收集的信息,整理并选择有足够垃圾的Region进行回收。这个过程会计算每个Region的回收收益,优先清理垃圾最多的Region。

混合收集(Mixed GC)

  • 混合GC:这是G1垃圾收集器的特性之一,结合了新生代和老年代的垃圾收集。混合收集不仅回收新生代(Eden和Survivor),同时还会选择性地回收一部分老年代的Region。
  • 标记-压缩:在混合GC步骤中,G1对老年代的回收通常使用的是标记-压缩(Mark-Compact)算法。标记阶段标记存活的对象,压缩阶段会移动对象来消除内存碎片,从而可以有连续的空间分配给新的大对象。

回收过程中的暂停时间和吞吐量

  • 暂停时间:G1垃圾收集器设计了暂停时间预测算法,可以根据用户配置的最大暂停时间(-XX:MaxGCPauseMillis)来适应性调整收集行为。
  • 吞吐量:尽量增加应用线程的运行时间,减少垃圾收集的总停顿时间。

总体流程

  1. 启动Young GC:当Eden区满时触发年轻代收集,使用复制算法将存活对象移动到Survivor区或老年代。
  2. 并发标记阶段启动:当老年代使用率超过特定阈值时,G1启动并发标记周期。
  3. 混合收集:在并发标记周期完成后,会进行多次混合收集,回收新生代和部分老年代。
  4. 完全垃圾收集(Full GC):作为最后的手段,当其他方法无法腾出足够空间时,G1会进行Full GC,这是代价最高的一种垃圾收集方式。

通过以上综合设计,G1垃圾收集器能够在大堆内存环境中有效地管理内存,降低垃圾收集停顿时间,提高应用的响应性和吞吐量。

3、是否可以理解:G1的内存管理 = CMS + 标记整理

标记-整理算法:在标记-整理算法中,存活对象会被移动以压缩和消除内存碎片,这个过程需要停止所有应用线程(“Stop-The-World”),导致较长的停顿时间。

虽然CMS可以通过引入标记-整理算法来减少碎片,但无法有效解决长时间停顿和并发处理的复杂性问题。G1通过划分Region+混合收集,使得G1能够更加灵活地选择收集哪些区域,而不是像传统的分代收集器那样对整个年轻代或老年代进行回收。

G1混合收集触发场景

  1. 周期性触发:G1有一个称为“Initiating Heap Occupancy Percent”(IHOP)的阈值,默认情况下,当整个Java堆的使用量达到这个阈值时(默认设置通常是45%到70%之间,可以通过-XX:InitiatingHeapOccupancyPercent参数调整),G1会启动一次混合收集。这意味着G1会定期检查堆的占用情况,一旦达到设定的阈值,就会触发一次收集以回收空间。

  2. 避免内存耗尽:除了基于IHOP的触发机制外,G1还会根据实际的内存分配速率和当前堆的使用情况动态调整,确保有足够的空闲空间来满足应用程序的内存需求,避免因为没有足够的空间分配对象而导致的Full GC。

  3. 目标停顿时间管理:G1在执行时会尽量维持用户设定的目标停顿时间(通过-XX:MaxGCPauseMillis参数指定)。如果系统检测到为了满足这个目标停顿时间,需要进行老年代的清理,那么也会触发混合收集,即使没有达到IHOP设定的阈值。

  4. 碎片整理需求:G1在运行过程中会持续监控堆的碎片化程度。如果发现老年代区域过于碎片化,影响了对象分配或者满足不了预设的内存分配需求,G1也可能决定进行一次混合收集来整理内存,减少碎片。

混合收集的主要目的是回收一部分老年代区域,同时保持暂停时间的可预测性,并通过这种方式平衡吞吐量与响应时间的需求。在整个过程中,G1会优先回收垃圾最多(即回收效益最高的)区域,以提高收集效率。

4、为什么选择G1

JDK 9之后,G1(Garbage First)垃圾收集器成为了默认的垃圾收集器。这一改变主要基于以下几点考虑:

  1. 适应性更强:G1是一种跨代收集器,能够自动管理整个Java堆,包括年轻代和老年代,而不仅仅是老年代(如同CMS)。

  2. 更低暂停、更高吞吐:通过并发和并行处理(充分利用多核处理器的能力),以及灵活的Region管理,减少了应用的全局停顿时间,适合对响应时间要求较高的应用。

  3. 内存管理更高效:通过标记-压缩算法和区域回收,有效地管理内存碎片,避免大对象分配困难的问题。同时引入了混合收集模式,即在年轻代收集的同时,也会并发地回收一部分老年代,这有助于减少老年代的碎片化问题。

参考:

  • https://juejin.cn/post/6844903893906751501
  • https://docs.oracle.com/javase/9/gctuning/garbage-first-garbage-collector.htm#JSGCT-GUID-ED3AB6D3-FD9B-4447-9EDF-983ED2F7A573

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

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

相关文章

一体化运维监控平台:赋能各行业用户运维升级

在当今数字化转型的大潮中,企业IT系统的复杂性和规模不断攀升,对运维团队提出了前所未有的挑战。如何高效、精准地监控和管理IT基础设施,确保业务连续性和稳定性,成为所有企业关注的焦点。美信,自2007年成立以来&#…

el-scrollbar实现自动滚动到底部(AI聊天)

目录 项目背景 实现步骤 实现代码 完整示例代码 项目背景 chatGPT聊天消息展示滚动面板,每次用户输入提问内容或者ai进行流式回答时需要不断的滚动到底部确保展示最新的消息。 实现步骤 采用element ui 的el-scrollbar作为聊天消息展示组件。 通过操作dom来实…

端、边、云三级算力网络

目录 端、边、云三级算力网络 NPU Arm架构 OpenStack kubernetes k3s轻量级Kubernetes kubernetes和docker区别 DCI(Data Center Interconnect) SD/WAN TF 端、边、云三级算力网络 算力网络从传统云网融合的角度出发,结合 边缘计算、网络云化以及智能控制的优势,通…

Qt开发 | Qt创建线程 | Qt并发-QtConcurrent

文章目录 一、Qt创建线程的三种方法二、Qt并发:QtConcurrent介绍三、QtConcurrent run参数说明四、获取QtConcurrent的返回值五、C其他线程技术介绍 一、Qt创建线程的三种方法 以下是Qt创建线程的三种方法: 方法一:派生于QThread 派生于QThre…

理解算法复杂度:空间复杂度详解

引言 在计算机科学中,算法复杂度是衡量算法效率的重要指标。时间复杂度和空间复杂度是算法复杂度的两个主要方面。在这篇博客中,我们将深入探讨空间复杂度,了解其定义、常见类型以及如何进行分析。空间复杂度是衡量算法在执行过程中所需内存…

ceph mgr [errno 39] RBD image has snapshots (error deleting image from trash)

ceph mgr 报错 debug 2024-07-08T09:25:56.512+0000 7f9c63bd2700 0 [rbd_support INFO root] execute_task: task={"sequence": 3, "id": "260b9fee-d567-4301-b7eb-b1fe1b037413", "message": "Removing image replicapool/8…

昇思25天学习打卡营第19天|Diffusion扩散模型

学AI还能赢奖品?每天30分钟,25天打通AI任督二脉 (qq.com) Diffusion扩散模型 本文基于Hugging Face:The Annotated Diffusion Model一文翻译迁移而来,同时参考了由浅入深了解Diffusion Model一文。 本教程在Jupyter Notebook上成…

python库 - missingno

missingno 是一个用于可视化和分析数据集中缺失值的 Python 库。它提供了一系列简单而强大的工具,帮助用户直观地理解数据中的缺失模式,从而更好地进行数据清洗和预处理。missingno 库特别适用于数据分析和数据科学项目,尤其是在处理缺失数据…

昇思MindSpore学习笔记5-02生成式--RNN实现情感分类

摘要: 记录MindSpore AI框架使用RNN网络对自然语言进行情感分类的过程、步骤和方法。 包括环境准备、下载数据集、数据集加载和预处理、构建模型、模型训练、模型测试等。 一、概念 情感分类。 RNN网络模型 实现效果: 输入: This film is terrible 正…

放大镜案例

放大镜 <!DOCTYPE html> <html lang"zh-cn"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>商品放大镜</title><link rel&qu…

如何使用allure生成测试报告

第一步下载安装JDK1.8&#xff0c;参考链接JDK1.8下载、安装和环境配置教程-CSDN博客 第二步配置allure环境&#xff0c;参考链接allure的安装和使用(windows环境)_allure windows-CSDN博客 第三步&#xff1a; 第四步&#xff1a; pytest 查看目前运行的测试用例有无错误 …

如何使用 pytorch 创建一个神经网络

我已发布在&#xff1a;如何使用 pytorch 创建一个神经网络 SapientialM.Github.io 构建神经网络 1 导入所需包 import os import torch from torch import nn from torch.utils.data import DataLoader from torchvision import datasets, transforms2 检查GPU是否可用 dev…

ffmpeg滤镜创建过程

1、创建一个滤镜图 AVFilterGraph *filter_graph avfilter_graph_alloc(); 2、创建滤镜的输入和输出 AVFilterInOut *inputs avfilter_inout_alloc(); AVFilterInOut *outputs avfilter_inout_alloc(); 3、每个滤镜创建上下文 AVFilterContext *filter1_ctx avfilter_…

Yolov10训练,转化onnx,推理

yolov10对于大目标的效果好&#xff0c;小目标不好 一、如果你训练过yolov5&#xff0c;yolov8&#xff0c;的话那么你可以直接用之前的环境就行 目录 一、如果你训练过yolov5&#xff0c;yolov8&#xff0c;的话那么你可以直接用之前的环境就行 二、配置好后就可以配置文件…

android webview 远程调试

打开远程调试选项 MainActivity super.onCreate(savedInstanceState);// enable Cordova apps to be started in the backgroundBundle extras getIntent().getExtras();if (extras ! null && extras.getBoolean("cdvStartInBackground", false)) {moveT…

前端JS特效第24集:jquery css3实现瀑布流照片墙特效

jquery css3实现瀑布流照片墙特效&#xff0c;先来看看效果&#xff1a; 部分核心的代码如下(全部代码在文章末尾)&#xff1a; <!DOCTYPE html> <html lang"en"> <head> <meta charset"UTF-8" /> <title>jquerycss3实现瀑…

Nginx:负载均衡小专题

运维专题 Nginx&#xff1a;负载均衡小专题 - 文章信息 - Author: 李俊才 (jcLee95) Visit me at CSDN: https://jclee95.blog.csdn.netMy WebSite&#xff1a;http://thispage.tech/Email: 291148484163.com. Shenzhen ChinaAddress of this article:https://blog.csdn.net/…

在Conda环境中高效使用Kubernetes:跨平台容器化实践指南

摘要 Conda 是一个流行的跨平台包和环境管理器&#xff0c;广泛用于Python社区。而 Kubernetes 是一个开源的容器编排系统&#xff0c;用于自动化部署、扩展和管理容器化应用程序。本文将探讨如何在 Conda 环境中使用 Kubernetes&#xff0c;包括设置 Conda 环境、容器化应用程…

【专项刷题】— 位运算

常见类型介绍&#xff1a; & &#xff1a;有 0 就是 0 | &#xff1a;有 1 就是 1 ^ &#xff1a;相同为 0 &#xff0c;相异为 1 或者 无进位相加给定一个数确定它的二进制位的第x个数是0还是1&#xff1a;将一个数的二进制的第x位改成1&#xff1a;将一个数的二进制的第x…

Windows10/11家庭版开启Hyper-V虚拟机功能详解

Hyper-V是微软的一款虚拟机软件&#xff0c;可以使我们在一台Windows PC上&#xff0c;在虚拟环境下同时运行多个互相之间完全隔离的操作系统&#xff0c;这就实现了在Windows环境下运行Linux以及其他OS的可能性。和第三方虚拟机软件&#xff0c;如VMware等相比&#xff0c;Hyp…