jvm垃圾收集器之七种武器

目录

1.回收算法

1.1 标记-清除算法(Mark-Sweep)

1.2 复制算法(Copying)

1.3 标记-整理算法(Mark-Compact)

2.HotSpot虚拟机的垃圾收集器

2.1 新生代的收集器

Serial 收集器(复制算法)

ParNew 收集器 (复制算法)

Parallel Scavenge 收集器 (复制算法)

2.2 老年代的收集器

Serial Old 收集器 (标记-整理算法)

Parallel Old 收集器 (标记-整理算法)

CMS(Concurrent Mark Sweep)收集器(标记-清除算法)

2.3 整个堆

G1(Garbage First)收集器 (标记-整理算法)


1.回收算法

1.1 标记-清除算法(Mark-Sweep)

分为两个阶段,标注和清除。标记阶段标记出所有需要回收的对象,清除阶段回收被标记的对象所占用的空间。

该算法最大的问题是内存碎片化严重,后续可能发生大对象不能找到可利用空间的问题。

1.2 复制算法(Copying)

按内存容量将内存划分为等大小的两块。每次只使用其中一块,当这一块内存满后将尚存活的对象复制到另一块上去,把已使用的内存清掉。

这种算法虽然实现简单,内存效率高,不易产生碎片,但是最大的问题是可用内存被压缩到了原本的一半。且存活对象增多的话,Copying算法的效率会大大降低。

1.3 标记-整理算法(Mark-Compact)

标记后不是清理对象,而是将存活对象移向内存的一端。然后清除端边界外的对象。

2.HotSpot虚拟机的垃圾收集器

展示了7种作用于不同分代的收集器,如果两个收集器之间存在连线,就说明它们可以搭配使用。虚拟机所处的区域,则表示它是属于新生代收集器还是老年代收集器。

2.1 新生代的收集器

Serial 收集器(复制算法)

新生代单线程收集器,标记和清理都是单线程,

优点:

简单高效:对于有限的资源环境,Serial 收集器由于其简单的算法和单线程的执行方式,在垃圾收集上能够达到一定的效率。

易于实现:没有多线程间同步的复杂性,使得Serial 收集器相对容易实现和维护。

缺点:

停顿时间:所有工作线程在垃圾收集时都需要暂停,这可能导致应用响应时间变长,特别是堆内存较大时,GC停顿时间会更加明显。

不适合多核处理器:在多核处理器系统上,由于Serial 收集器只能使用单个核心进行垃圾收集,无法充分利用现代硬件资源。

Serial/Serial Old收集器运行示意图

ParNew 收集器 (复制算法)

新生代并行收集器,实际上是 Serial 收集器的多线程版本

使用-XX:+UseParNewGC(新生代使用并行收集器,老年代使用串行回收收集器) 或者-XX:+UseConcMarkSweepGC(新生代使用并行收集器,老年代使用 CMS)

优点:

  1. 多线程收集:ParNew 收集器能够并行地使用多个线程进行垃圾收集,这使得它在多核处理器上相较于 Serial 收集器具有更好的性能表现。
  2. 适合多核环境:随着现代服务器通常配备多核处理器,ParNew 收集器能够更有效地利用这些处理器资源,提高垃圾收集的效率。
  3. 与CMS兼容:ParNew 收集器经常与老年代的 Concurrent Mark Sweep (CMS) 收集器配合使用,为需要低停顿时间且具有多核处理器的应用提供了一种有效的垃圾收集解决方案。
  4. 适用于交互式应用:由于 ParNew 收集器能够减少垃圾收集时的停顿时间,它特别适合于对响应时间有较高要求的交互式应用。

缺点:

  1. 消耗更多的CPU资源:虽然 ParNew 收集器通过并行收集减少了停顿时间,但这也意味着在垃圾收集过程中会使用更多的CPU资源,可能会对应用程序的其他部分造成影响。
  2. 配置和调优复杂:ParNew 收集器的配置和调优相对于 Serial 收集器更加复杂,需要合理设置并行线程数等参数以达到最佳性能。
  3. 停顿时间仍存在:尽管 ParNew 收集器减少了垃圾收集的停顿时间,但在进行垃圾收集时,仍然需要暂停所有工作线程(Stop-The-World),在大堆内存的情况下,这可能仍然导致明显的停顿。
  4. 在JDK 9+中被淘汰:随着G1收集器的引入和成为默认的垃圾收集器,ParNew 加 CMS 的组合在 JDK 9 及之后的版本中不再是主流选择,因为 G1 提供了更平衡的吞吐量和停顿时间,以及更简单的调优过程。

ParNew/Serial Old收集器运行示意图

Parallel Scavenge 收集器 (复制算法)

新生代并行收集器, 追求高吞吐量, 高效利用 CPU

-XX:MaxGCPauseMillis 配置最大垃圾收集停顿时间 -XX:GCTimeRatio 配置吞吐量大小

并行垃圾回收器在进行垃圾回收时, 同样会持有所有应用程序的线程, 并冻结所有应用程序线程,来进行垃圾回收工作

优点:

  1. 高吞吐量:Parallel Scavenge 收集器注重达到一个可控制的吞吐量(Throughput),它使用多个线程并行执行垃圾收集操作,可以充分利用多核CPU的计算能力,从而实现高吞吐量的垃圾收集。这使得它非常适合处理大规模数据和对处理能力要求较高的应用。
  2. 快速回收:该收集器主要关注减少垃圾收集时的停顿时间,通过将垃圾收集任务划分为多个阶段,并使用多个线程并行执行,可以更快地完成垃圾收集,并尽量减少应用程序的暂停时间。
  3. 可控制的吞吐量和最大停顿时间:Parallel Scavenge 收集器提供了一些参数来精确控制吞吐量和最大停顿时间的平衡。可以根据应用程序的性能需求来调整这些参数,以达到最佳的垃圾收集性能。

缺点:

  1. 高延迟:由于 Parallel Scavenge 收集器注重吞吐量而非低延迟,因此在追求高吞吐量的同时,可能会导致垃圾收集的停顿时间相对较长。这对于某些对实时性要求较高的应用程序可能不太适合。
  2. 内存占用:Parallel Scavenge 收集器为了追求高吞吐量,通常会使用较大的堆空间来存储对象,并且不会立即释放未使用的内存。这可能导致在某些情况下,垃圾收集器无法及时回收所有可回收的内存,从而造成一定的内存浪费。

Parallel Scavenge/Parallel Old收集器运行示意图

2.2 老年代的收集器

Serial Old 收集器 (标记-整理算法)

老年代单线程收集器,Serial 收集器的老年代版本,使用“标记-整理”(Mark-Compact)算法来进行垃圾收集。在标记阶段,它会遍历堆内存,标记出所有可达的对象;然后在整理阶段,将所有存活的对象向一端移动,从而清理出连续的空闲内存空间,减少内存碎片。

优点:

  1. 简单高效:对于单核处理器或者小内存资源的应用,Serial Old 收集器因为其简单和直接的垃圾回收方式,在这类环境下可以非常高效。
  2. 易于实现和调试:由于只涉及单线程操作,使得 Serial Old 收集器相对容易实现和调试。

缺点:

  1. 停顿时间长:所有应用线程都必须在垃圾收集期间暂停,这可能导致较长的停顿时间,影响到应用的响应速度和吞吐量。
  2. 不适合多核处理器:在多核处理器系统上,Serial Old 收集器无法充分利用硬件资源,因此并不适合大型、多线程的服务器端应用。

Serial/Serial Old收集器运行示意图

Parallel Old 收集器 (标记-整理算法)

老年代并行收集器, 吞吐量优先, Parallel Scavenge收集器的老年代版本

特点

  1. 多线程:Parallel Old 收集器同样采用多线程进行垃圾回收,这意味着它可以充分利用多核心处理器的优势,提高垃圾收集的效率。
  2. 算法:它使用“标记-整理”(Mark-Compact)算法来处理老年代的垃圾收集。与“标记-清除”(Mark-Sweep)算法相比,“标记-整理”算法在完成垃圾收集后不会留下空间碎片,从而避免了长时间运行后可能出现的内存分配问题。
  3. 适用场景:Parallel Old 收集器适用于多核服务器环境中,对吞吐量有较高要求的场景。它能够提供比CMS更好的吞吐量,但在停顿时间上可能不如CMS收集器和G1收集器

Parallel Scavenge/Parallel Old收集器运行示意图

CMS(Concurrent Mark Sweep)收集器(标记-清除算法)

收集器是一种以获取最短回收停顿时间为目标的垃圾收集器,主要用于收集老年代的垃圾。它适用于对响应时间有较高要求的场景,通过并发标记和清除来实现减少停顿时间的目的。

整个过程分为4个步骤,包括:

1.初始标记(CMS initial mark

  • 特点:这是一个需要“Stop-The-World”(STW)的阶段,但其持续的时间相对较短。
  • 工作内容:标记所有直接与GC Roots相连的对象,以及从年轻代到老年代的引用对象(跨代引用)。此阶段虽然停顿时间短,但需要暂停所有应用线程。

2.并发标记(CMS concurrent mark)

  • 特点:在这个阶段,GC线程和应用线程可以并发运行,不需要暂停应用线程。
  • 工作内容:从初始标记阶段找到的根对象开始遍历整个对象图,标记所有可达的存活对象。由于在并发标记过程中应用线程仍在运行,可能会改变对象引用关系,因此需要记录这些改变。

3.重新标记(CMS remark)

  • 特点:这也是一个需要STW的阶段,但CMS采用了多种技术(如增量更新、原始快照等)来缩短这一阶段的停顿时间。
  • 工作内容:修正并发标记阶段因程序运行导致的标记记录变动。为了减少停顿时间,CMS在这个阶段使用了算法,如增量更新(Incremental Update)和原始快照(SATB,Snapshot At The Beginning),来处理并发阶段遗留的问题。

4.并发清除(CMS concurrent sweep)

  •  特点:应用线程和垃圾收集线程可以并发执行,不需要暂停应用线程。
  • 工作内容:清理掉在标记阶段标记为已死亡的对象,并回收他们占用的内存空间。这一阶段结束后,CMS收集器将已经清理好的内存空间归还给JVM使用。

但是CMS还远达不到完美的程度,它有以下3个明显的缺点:

  • CMS收集器对CPU资源非常敏感。
  • CMS收集器无法处理浮动垃圾(Floating Garbage),
  • CMS收集结束时会有大量空间碎片产生。

Concurrent Mark Sweep收集器运行示意图

2.3 整个堆

G1(Garbage First)收集器 (标记-整理算法)

G1 回收的范围是整个 Java 堆(包括新生代,老年代)

G1具备如下特点。

  • 并行与并发:G1能充分利用多CPU、多核环境下的硬件优势,使用多个CPU(CPU或者CPU核心)来缩短Stop-The-World停顿的时间,部分其他收集器原本需要停顿Java线程执行的GC动作,G1收集器仍然可以通过并发的方式让Java程序继续执行。
  • 分代收集:与其他收集器一样,分代概念在G1中依然得以保留。虽然G1可以不需要其他收集器配合就能独立管理整个GC堆,但它能够采用不同的方式去处理新创建的对象和已经存活了一段时间、熬过多次GC的旧对象以获取更好的收集效果。
  • 空间整合:与CMS的“标记—清理”算法不同,G1从整体来看是基于“标记—整理”算法实现的收集器,从局部(两个Region之间)上来看是基于“复制”算法实现的,但无论如何,这两种算法都意味着G1运作期间不会产生内存空间碎片,收集后能提供规整的可用内存。这种特性有利于程序长时间运行,分配大对象时不会因为无法找到连续内存空间而提前触发下一次GC。
  • 可预测的停顿:这是G1相对于CMS的另一大优势,降低停顿时间是G1和CMS共同的关注点,但G1除了追求低停顿外,还能建立可预测的停顿时间模型,能让使用者明确指定在一个长度为M毫秒的时间片段内,消耗在垃圾收集上的时间不得超过N毫秒,这几乎已经是实时Java(RTSJ)的垃圾收集器的特征了。

G1收集器的运作大致可划分为以下几个步骤:

1.初始标记(Initial Marking)

  • 特点:此阶段需要暂停所有的应用线程(Stop-The-World, STW),但持续时间相对较短。
  • 工作内容:标记从GC Roots直接可达的对象,以及年轻代中存活对象的边界。

2.根区域扫描(Root Region Scanning)

  • 特点:在应用线程运行的同时执行,不需要暂停应用线程。
  • 工作内容:处理初始标记阶段找到的存活对象所引用的对象。这一步骤必须在年轻代的下一次垃圾收集前完成。

3.并发标记(Concurrent Marking)

  •  特点:与应用线程并发执行,不会导致应用线程停顿。
  • 工作内容:从根区域扫描阶段标记的对象开始,遍历整个堆,标记出所有可达的存活对象。

4.最终标记(Final Marking)

  • 特点:需要STW,但通过优化(如使用记忆集和卡表来减少标记范围)尽量缩短停顿时间。
  • 工作内容:处理并发标记阶段遗留的少量工作,如处理SATB(Snapshot At The Beginning)队列中的剩余对象,以及修正并发标记期间因应用程序运行产生的变动。

5.筛选回收(Live Data Counting and Evacuation)

  •  特点:需要STW,此阶段的停顿时间是G1收集器控制的重点。
  • 工作内容:根据之前的标记结果,选择一部分内存区域进行清理。G1会优先选择回收价值最大的区域(即垃圾最多的区域)进行清理,以提高垃圾收集的效率。在这个阶段,存活的对象会被移动到其他区域,同时整理碎片,释放空间。

G1收集器运行示意图

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

OpenCV-32 膨胀操作

膨胀是与腐蚀相反的操作,基本原理是只要保证卷积核的锚点是非0值,周边无论是0还是非0值,都变为0。 使用API---dilate(img, kernel, iterationms 1) 示例代码如下: import cv2 imp…

C# 实现微信自定义分享

目录 需求与调整 代码实现 获取令牌 生成合法票据 获取有效签名 客户端准备 客户端实现 小结 需求与调整 在微信中打开网页应用后,可以选择将地址发送给朋友进行分享,如下图: 在实际的应用中,我们可能不是简单的将该网页…

HiveSQL——借助聚合函数与case when行转列

一、条件函数 if 条件函数 if函数是最常用到的条件函数&#xff0c;其写法是if(xn,a,b), xn代表判断条件&#xff0c;如果xn时&#xff0c;那么结果返回a ,否则返回b。 selectif(age < 25 or age is null, 25岁以下, 25岁以上) as age_cnt,count(1) as number from table…

搭建macOS开发环境-1:准备工作

请记住&#xff1a; 最重要的准备工作永远是&#xff1a;备份数据 !!! 通过图形界面检查 Mac 的 CPU 类型&#xff1a; 在搭载 Apple 芯片的 Mac 电脑上&#xff0c;“关于本机”会显示一个标有“芯片”的项目并跟有相应芯片的名称&#xff1a; 通过命令行检查Mac的CPU类型 …

java_error_in_pycharm.hprof文件是什么?能删除吗?

java_error_in_pycharm.hprof文件是什么&#xff1f;能删除吗&#xff1f; &#x1f335;文章目录&#x1f335; &#x1f333;引言&#x1f333;&#x1f333;hprof格式文件介绍&#x1f333;&#x1f333;java_error_in_pycharm.hprof文件什么情况下能删除&#x1f333;&…

Vue CLI学习笔记

在看任何开源库的源码之前&#xff0c;必须先了解它有哪些功能&#xff0c;这样才能针对性地分模块阅读源码。 Vue CLI 简介 Vue CLI是Vue.js的官方命令行工具&#xff0c;它是一个基于Vue.js进行快速开发的完整系统。 通过Vue CLI&#xff0c;开发者可以快速搭建和开发Vue.js项…

rsyslog远程记录系统日志

rsyslog是一个快速处理手机系统日志的开源程序&#xff0c;提供了高性能&#xff0c;安全功能和模块化设计&#xff0c;rsyslog是syslog的升级版&#xff0c;他讲多重来源输入输出转换结果到目的地&#xff0c;rsyslog被广泛用于Linux系统以通过TCP/UDP协议转发或接收日志消息。…

Gitlab和Jenkins集成 实现CI (一)

版本声明 部署时通过docker拉取的最新版本 gitlab: 16.8 jenkins: 2.426.3 安装环境 可参考这篇文章 停止防火墙 由于在内网&#xff0c;这里防火墙彻底关掉&#xff0c;如果再外网或者云上的悠着点 systemctl stop firewalled systemctl disable firewalledsystemctl sto…

林浩然的趣味解读:赫伯特·亚历山大·西蒙的跨界智慧与伟大成就

林浩然的趣味解读&#xff1a;赫伯特亚历山大西蒙的跨界智慧与伟大成就 Lin Haoran’s Amusing Interpretation: Herbert Alexander Simon’s Interdisciplinary Wisdom and Great Achievements 林浩然&#xff0c;这位机智幽默且对知识充满好奇的探索者&#xff0c;最近在一次…

深入探索:缓冲区溢出漏洞及其防范策略

在网络安全的广阔领域中&#xff0c;缓冲区溢出漏洞一直是一个重要的议题。这种漏洞&#xff0c;如果被恶意利用&#xff0c;可能会导致严重的安全问题&#xff0c;包括数据泄露、系统崩溃&#xff0c;甚至可能被攻击者利用来执行恶意代码。在本文中&#xff0c;我们将深入探讨…

TCP相关知识点

TCP相关知识点 参考&#xff1a; 《计算机网络》 (建议收藏)TCP协议灵魂之问&#xff0c;巩固你的网路底层基础 关于 TCP 三次握手和四次挥手&#xff0c;满分回答在此 (值得看) TCP处于网络体系结构中的运输层。 运输层主要为应用进程提供端到端的逻辑通信&#xff0c;然后对…

RabbitMQ高可用架构涉及常用功能整理

RabbitMQ高可用架构涉及常用功能整理 1. rabbitmq的集群模式2. 镜像模式高可用系统架构和相关组件3. rabbitmq的核心参数3.1 镜像策略3.2 新镜像同步策略3.3 从节点晋升策略3.4 主队列选择策略 4. rabbitmq常用命令4.1 常用基础命令4.1.1 服务管理4.1.2 用户管理4.1.3 角色管理…

机器学习--K-近邻算法常见的几种距离算法详解

文章目录 距离度量1 欧式距离(Euclidean Distance)2 曼哈顿距离(Manhattan Distance)3 切比雪夫距离 (Chebyshev Distance)4 闵可夫斯基距离(Minkowski Distance)5 标准化欧氏距离 (Standardized EuclideanDistance)6 余弦距离(Cosine Distance)7 汉明距离(Hamming Distance)【…

springboot154基于Spring Boot智能无人仓库管理

简介 【毕设源码推荐 javaweb 项目】基于springbootvue 的 适用于计算机类毕业设计&#xff0c;课程设计参考与学习用途。仅供学习参考&#xff0c; 不得用于商业或者非法用途&#xff0c;否则&#xff0c;一切后果请用户自负。 看运行截图看 第五章 第四章 获取资料方式 **项…

【Linux】vim的基本操作与配置(上)

Hello everybody!今天我们要进入vim的讲解了。学会了vim,咱们就可以在Linux系统上做一些简单的编程啦&#xff01; 那么废话不多说&#xff0c;咱们直接进入正题&#xff01; 1.初识vim vim是一款多模式的文本编辑器&#xff0c;可以对一个文件进行编辑操作。 它一共有三个模…

Verilog刷题笔记23

题目: Suppose you’re building a circuit to process scancodes from a PS/2 keyboard for a game. Given the last two bytes of scancodes received, you need to indicate whether one of the arrow keys on the keyboard have been pressed. This involves a fairly simp…

【高阶数据结构】B-树详解

文章目录 1. 常见的搜索结构2. 问题提出使用平衡二叉树搜索树的缺陷使用哈希表的缺陷 3. B-树的概念4. B-树的插入分析插入过程分析插入过程总结 5. B-树的代码实现5.1 B-树的结点设计5.2 B-树的查找5.3 B-树的插入实现InsertKey插入和分裂测试 6. B-树的删除&#xff08;思想&…

牛客网SQL进阶137:第二快/慢用时之差大于试卷时长一半的试卷

官网链接&#xff1a; 第二快慢用时之差大于试卷时长一半的试卷_牛客题霸_牛客网现有试卷信息表examination_info&#xff08;exam_id试卷ID, tag试卷类别,。题目来自【牛客题霸】https://www.nowcoder.com/practice/b1e2864271c14b63b0df9fc08b559166?tpId240 0 问题描述 试…

【Git】08 多人单分支协作场景

文章目录 一、场景1&#xff1a;不同人修改不同文件1.1 场景描述1.2 场景复现1.2.1 克隆到本地1.2.2 新建分支1.2.3 B修改、提交与推送1.2.4 A修改与提交1.2.5 B再次修改并推送1.2.6 A推送报错 1.3 解决 二、场景2&#xff1a;不同人修改同文件的不同区域2.1 场景描述2.2 场景复…

iOS 需求 多语言(国际化)App开发 源码

一直觉得自己写的不是技术&#xff0c;而是情怀&#xff0c;一个个的教程是自己这一路走来的痕迹。靠专业技能的成功是最具可复制性的&#xff0c;希望我的这条路能让你们少走弯路&#xff0c;希望我能帮你们抹去知识的蒙尘&#xff0c;希望我能帮你们理清知识的脉络&#xff0…