Java高阶私房菜:JVM分代收集算法介绍和各垃圾收集器原理分解

news2025/1/11 18:45:26

目录

什么是分代收集算法

GC的分类和专业术语

什么是垃圾收集器

垃圾收集器的分类及组合

​编辑

应关注的核心指标

Serial和ParNew收集器原理

Serial收集器

 ParNew收集器

Parallel和CMS收集器原理

Parallel 收集器

CMS收集器

新一代垃圾收集器G1和ZGC

G1垃圾收集器

回收原理讲解

三种垃圾回收模式

MixGC垃圾回收步骤

常见参数讲解

配置注意事项

ZGC-新一代垃圾搜集器

什么是ZGC

工作流程

平台支持说明

验证参数(JDK17环境)  

关于升级建议


        现在市面上充斥着许多对于JVM垃圾收集器教程,有零基础的,有专讲的 ... 这些教程或文章大都长篇大论的细扣字眼,内容繁多且没有重点,看过便会忘,本篇集结了9年老JAVA工作及面试经验,专讲面试和工作中使用到的,一站式讲解,帮助你疏通学习脉络,少走弯路。

学习本章节前,对于垃圾回收机制还不了解的话可以看看我的上篇文章进行回顾。传送门:Java高阶私房菜:JVM垃圾回收机制及算法原理探究-CSDN博客

什么是分代收集算法

        针对不同生命周期的对象采用不同的垃圾回收策略,以达到更好的垃圾回收效果。年轻代空间多数对象存活时间短,可以高频地进行回收;而在老年代空间多数对象存活时间久,则需要低频回收。分代算法是根据回收对象的特点进行选择,根据上篇总结,轻代适合标记-复制算法,老年代适合标记清除或标记压缩算法,通过将内存划分为不同的代,可以使得Minor GC的频率更高,更早地回收垃圾对象,减少Full GC的发生频率,提高整体性能。

JVM将内存分为年轻代和老年代,其中年轻代又分为Eden区和两个Survivor区。

        年轻代:用于存放新生的对象,其中Eden区是新对象的分配区域,当Eden区满时,会触发Minor GC,将存活的对象移动到Survivor区,同时清空Eden区,Survivor区则用于存放经过一次Minor GC后存活的对象。

        老年代:用于存放长时间存活的对象,当年轻代中的对象经过多次Minor GC后仍然存活,就会被移动到老年代,当老年代满时,会触发Full GC。

GC的分类和专业术语

Partical GC(部分收集)

        新生代收集:对象从Young 区域消失(被回收)的过程称为“minor GC / Young GC”,Eden 的清理,S0\S1的清理都由于MinorGC,当YoungGen区内存不足时,就会触发minorGC。

        老年代收集:对象从老年代中消失的过程,清理整合OldGen的内存空间,称为Major GC/Old GC,有些垃圾收集器 针对老年代单独回收,所以比较少用

Full GC(整堆收集)

        清理整个堆空间,包括年轻代和老年代,可以理解为Major GC+Minor GC组合后进行的一整个过程,是清理JVM整个堆空间。

其实不用过多关心是叫Major GC还是Full GC,应该关注当前的 GC是否停止了所有应用程序的线程,许多Major GC 是由 Minor GC 触发的, 出现 Major GC通常出现至少一次的Minor GCMajorGC 的速度一般要比Minor GC慢 10倍不止。

        其可手动调用System.gc( )触发, 但却不一定会立马执行Full GC,而只是提交了一个回收请求,由JVM的自动垃圾回收机制判断当前是否触发Full GC,即当老年代空间不足 , 通过Minor GC后进入老年代的平均大小大于老年代的可用内存

        需要注意的是,当执行Full GC时,它会暂停所有当前的应用线程,时间由回收效率而定,直至整个垃圾回收过程完成才恢复正常,它对系统的性能有较大影响,所以在生产环境中要尽量避免使用该方法,可以通过启动参数-XX:+ DisableExplicitGC来禁止调用System.gc()。内存的优化则尽量利用JVM垃圾回收策略来维护,包括日常的编码中不再使用的变量要置Null切断与堆空间的联系。

STW(Stop The World)

        垃圾回收发生过程中,用户线程在运行至安全点(safe point)后,就自行挂起进入暂停状态,对外的表现就是卡顿,所以应尽量减少Full GC的次数,是Minor GC还是Major GC都会STW,区别只在于STW的时间长短

JVM资料很多,有讲32位和64位的的虚拟机的参数,但是基本都不用32位操作系统(生产环境更是如此,如果仍使用32位操作系统就应该反省在这家公司的去留问题了),所以不用花时间去了解32位的。

什么是垃圾收集器

        还是那句话,垃圾回收算法是内存回收的方法论,垃圾收集器则是内存回收的具体实现,目前Java规范中并没有对垃圾收集器的实现有任何规范,不同的厂商、不同的版本的虚拟机提供的垃圾收集器是不同的,主要讨论的是HotSpot虚拟机。

那么有没有最厉害的垃圾收集器呢?负责任的说其实不存在最厉害的垃圾收集器,只有在对应场景中最合适的垃圾收集器。

那为什么要有这么多的收集器呢?那是因为Java的使用场景很多,移动端,服务器等,然后内存里面对象存活时间不一样,需要针对不同的场景,提供不同的垃圾收集器,提高垃圾收集的性能。

垃圾收集器的分类及组合

新生代收集器

  • Serial 串行垃圾收集器

  • ParNew 年轻代的并行垃圾回收器

  • Parallel 并行垃圾收集器

老年代收集器

  • Serial Old 串行老年代垃圾器

  • Parallel Old 老年代的并行垃圾回收器

  • CMS (ConcMarkSweep)并发标记清除

整堆收集器:

        G1:JDK9的默认垃圾收集器,以替代CMS。该收集器指出新生代和老年代不再是物理隔离的了(但还保留着新生代和老年代的概念),它们都是一部分Region (不需要连续)的集合,通过Region的动态分配方式实现逻辑上的连续。

        ZGC:JDK11新引入的ZGC收集器,在该收集器中不管是物理上还是逻辑上,已经不存在新老年代的概念了,它们会被分为一个个page,当进行GC操作时会对page进行压缩,因此没有碎片问题。

那么那么多垃圾收集器,我们是否都要记住?作为新时代互联网人,还是那句话,我们只学有用的,淘汰的垃圾收集器我们不花太多时间。

垃圾收集器的组合

        两个垃圾收集器之间如果存在连线,则说明它们可以搭配使用,图中:Serial old作为CMS出现"Concurrent Mode Failure"失败的后备预案

这里需要注意的是:

  • JDK8中默认使用: Parallel Scavenge GC + ParallelOld GC

  • JDK14 弃用了: Parallel Scavenge GC + Parallel OldGC

  • JDK9默认是用G1为垃圾收集器

  • JDK14 移除了 CMS GC

查看默认垃圾收集器

JVM参数: -XX:+PrintCommandLineFlags 查看命令行相关参数(包含使用的垃圾收集器)

设置默认垃圾收集器

// JDK8

  -XX:InitialHeapSize=536870912 -XX:MaxHeapSize=8589934592 -XX:+PrintCommandLineFlags -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseParallelGC 


// JDK11

  -XX:G1ConcRefinementThreads=9 -XX:GCDrainStackTargetSize=64 -XX:InitialHeapSize=536870912 -XX:MaxHeapSize=8589934592 -XX:+PrintCommandLineFlags -XX:ReservedCodeCacheSize=251658240 -XX:+SegmentedCodeCache -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseG1GC 


// JDK17

  -XX:ConcGCThreads=2 -XX:G1ConcRefinementThreads=9 -XX:GCDrainStackTargetSize=64 -XX:InitialHeapSize=536870912 -XX:MarkStackSize=4194304 -XX:MaxHeapSize=8589934592 -XX:MinHeapSize=6815736 -XX:+PrintCommandLineFlags -XX:ReservedCodeCacheSize=251658240 -XX:+SegmentedCodeCache -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseG1GC 

应关注的核心指标

吞吐量:运行用户代码的时间占总运行时间的比例(总运行时间 = 程序的运行时间 + 内存回收的时间),例如:虚拟机共运行100分钟,垃圾收集器花掉1分钟,那么吞吐量就是99%。

暂停时间:执行垃圾收集时,程序的工作线程被暂停的时间,一个时间段内应用程序线程暂停,让GC线程执行的状态,GC期间100毫秒的暂停时间,说明在这100毫秒期间内没有应用程序线程是活动的。

收集频率:指垃圾回收器多长时间会运行一次。一般来说,垃圾回收器的频率应该是越低越好。 

Serial和ParNew收集器原理

Serial收集器

        它作为最简单的垃圾收集器,使用单线程进行垃圾收集,暂停所有应用程序线程, 在单核CPU环境来说,Serial收集器是更高效的Serial Old是Serial收集器的老年代版本,在jdk1.5之前的版本与Parallel收集器搭配使用,或者作为CMS的备选方案。它适用于小型应用程序和客户端应用程序,一般javaweb、springboot项目不会采用这类收集器。

其实现算法视区域的不同:新生代采用复制算法,老年代采用标记整理算法

相关命令参数使用

  • 同时指定年轻代和老年代都使用串行垃圾收集器 -XX:+UseSerialGC

  • 查看命令行相关参数 -XX:+PrintCommandLineFlags

  • 全部参数 (JDK11环境)

//参数
-XX:+UseSerialGC -XX:+PrintCommandLineFlags -Xms32m -Xmx32m

//输出
-XX:InitialHeapSize=33554432 -XX:MaxHeapSize=33554432 -XX:+PrintCommandLineFlags -XX:ReservedCodeCacheSize=251658240 -XX:+SegmentedCodeCache -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseSerialGC 

 ParNew收集器

        工作在年轻代上的,只是将串行的垃圾收集器改为了并行,其他基本和Serial一样,使用多个线程进行垃圾回收的,适用于大型应用程序和多核处理器,以及在服务端应用程序中使用,单核上效率比Serial低,和下集讲Parallel收集器类似,但Parallel收集器不兼容CMS,除了它只有Serial收集器可以和CMS收集器配合工作。

其实现算法视区域的不同:新生代采用复制算法,老年代采用标记整理算法

相关命令参数使用

  • 年轻代使用ParNew回收器,老年代使用串行收集器 -XX:+UseParNewGC

  • 查看命令行相关参数 -XX:+PrintCommandLineFlags

  • 验证参数 (JDK8环境,如果用JDK11会报错,JDK8开始已经不再被推荐使用)

//参数
-XX:+UseParNewGC -XX:+PrintCommandLineFlags -Xms32m -Xmx32m

//输出  
-XX:InitialHeapSize=33554432 -XX:MaxHeapSize=33554432 -XX:+PrintCommandLineFlags -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseParNewGC 

Java HotSpot(TM) 64-Bit Server VM warning: Using the ParNew young collector with the Serial old collector is deprecated and will likely be removed in a future release

Parallel和CMS收集器原理

Parallel 收集器

        全称 Parallel Scavenge 是一种多线程垃圾收集器,和ParNew收集器类似,是一个新生代收集器默认线程数和cpu核数一样,用于大型应用程序和服务器应用程序,比如大批量数据处理,后台计算任务等,Parallel Old是Parallel Scavenge收集器的老年代版本,JDK8默认使用Parallel Scavenge收集器。

其实现算法视区域的不同:新生代采用复制算法,老年代采用标记整理算法

Parallel对比ParNew

  • -XX:+UseParallelGC 仅对年轻代有效,不可以和CMS收集同时使用

  • -XX:+UseParNewGC 设置年轻代为多线程收集,可以和CMS收集同时使用

相关命令参数使用

  • 年轻代使用ParallelGC垃圾回收器,老年代使用串行回收器 -XX:+UseParallelGC

  • 年轻代使用ParallelGC垃圾回收器,老年代使用ParallelOldGC垃圾回收器 -XX:+UseParallelOldGC

  • 查看命令行相关参数 -XX:+PrintCommandLineFlags

  • 验证参数(JDK11环境)

//参数一
-XX:+UseParallelGC -XX:+PrintCommandLineFlags -Xms32m -Xmx32m

//输出  
-XX:InitialHeapSize=33554432 -XX:MaxHeapSize=33554432 -XX:+PrintCommandLineFlags -XX:ReservedCodeCacheSize=251658240 -XX:+SegmentedCodeCache -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseParallelGC 

//参数二
-XX:+UseParallelGC -XX:+UseParallelOldGC -XX:+PrintCommandLineFlags -Xms32m -Xmx32m

//输出
-XX:InitialHeapSize=33554432 -XX:MaxHeapSize=33554432 -XX:+PrintCommandLineFlags -XX:ReservedCodeCacheSize=251658240 -XX:+SegmentedCodeCache -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseParallelGC -XX:+UseParallelOldGC 

CMS收集器

        全称 Concurrent Mark Sweep,是一款并发的、使用标记-清除算法的垃圾回收器,老年代中的对象生命周期较长,垃圾回收频率较低,目标是获取最短垃圾收集停顿时间,针对 老年代 垃圾的收集器,停顿时间较短,适合对响应时间要求较高的应用程序,如Web应用程序、电子商务等高并发场景。

        整个过程分4步(初始标记 和 重新标记 需要stopTheWorld,并发标记与并发清除阶段不需要暂停用户线程)

  • 初始标记: 标记GC Root直接关联对象,会导致stopTheWorld

  • 并发标记: 与用户线程同时运行

  • 重新标记:会导致stopTheWorld

  • 并发清除:与用户线程同时运行

相关命令参数使用

  • 年轻代使用ParNew垃圾回收器,老年代使用CMS回收器 -XX:+UseParNewGC -XX:+UseConcMarkSweepGC

  • 查看命令行相关参数 -XX:+PrintCommandLineFlags

  • 验证参数(JDK11环境,然后被弃用了,验证成功需要用JDK8)

//输入
-XX:+UseParNewGC -XX:+UseConcMarkSweepGC -XX:+PrintCommandLineFlags -Xms32m -Xmx32m

//输出
-XX:InitialHeapSize=33554432 -XX:MaxHeapSize=33554432 -XX:MaxNewSize=11190272 -XX:MaxTenuringThreshold=6 -XX:NewSize=11190272 -XX:OldSize=22364160 -XX:+PrintCommandLineFlags -XX:ReservedCodeCacheSize=251658240 -XX:+SegmentedCodeCache -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseConcMarkSweepGC 

//警告信息
 Option UseConcMarkSweepGC was deprecated in version 9.0 and will likely be removed in a future release.

上述垃圾收集器都是接近被弃用了,大体了解即可,重点掌握接下去的G1和ZGC

新一代垃圾收集器G1和ZGC

G1垃圾收集器

回收原理讲解

        Garbage First 垃圾收集器是JDK7版本之后引入的一种垃圾回收器,Jdk9中将G1变成默认的垃圾收集器,可以在不同的内存区域中分配垃圾回收的工作,提高了垃圾回收效率。

        它保留了分代思想,把内存划分为多个独立的区域Region,区域中包含逻辑上的年轻代、老年代区域。取消了年轻代、老年代的物理划分,不用单独对每个年代空间进行设置。Region的区域类型是动态变化的,可能之前是年轻代,经过了垃圾回收之后就变成了老年代,实现更加精细化的垃圾回收。整体采用标记整理算法, 局部是采用复制算法,不会产生内存碎片。它把整个Java堆划分成约2048个独立Region块,每个Region块大小根据堆空间的大小而定,为2的N次幂,1MB~32MB,即每个Region的大小可通过参数 -XX:G1HeapRegionSize 配置

        新增加一种叫Humongous内存区域,用于存储大对象,它规定如果超过1.5个region,就是巨型对象,就放到H区,默认直接会被分配在老年代,一般被视为老年代.图中的H块,如果一个H区装不下一个巨型对象,G1会寻找连续的H区来存储,为了能找到连续的H区,有时需要启动Full GC。

相关命令参数使用

  • 查看默认垃圾收集器-XX:+PrintCommandLineFlags 查看命令行相关参数(包含使用的垃圾收集器)

//JDK11

-XX:G1ConcRefinementThreads=9 -XX:GCDrainStackTargetSize=64 -XX:InitialHeapSize=536870912 -XX:MaxHeapSize=8589934592 -XX:+PrintCommandLineFlags -XX:ReservedCodeCacheSize=251658240 -XX:+SegmentedCodeCache -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseG1GC 

三种垃圾回收模式

Young GC

        G1与之前垃圾收集器的Young GC不同,不是当新生代的Eden区放满了就进行垃圾回收,G1会计算当前Eden区回收大概需要多久时间,如果接近参数-XX:MaxGCPauseMills设定的值,便会触发Young GC。

        回收过程也是将Eden区和Survivor区中的存活对象复制到空闲的Survivor区,并清空Eden区和原来的Survivor区。如果Survivor区也满了,那么会将存活对象复制到Old区,在Young GC期间,应用程序会被暂停。

Mixed GC

        多数对象晋升到老年代old region时,为了避免堆内存被耗尽问题,会触发混合的GC。回收整个Young Region,还会回收一部分的Old Region区域,注意不是全部Old Region区域。其触发条件由配置的参数 -XX:InitiatingHeapOccupancyPercent=n决定。默认为45%,即 当老年代大小占整个堆大小百分比达到该阀值时触发。

Full GC

        单个线程会对整个堆的所有代中所有分区做标记、清除以及压缩动作,非常耗时。

总结

        在Young GC和Mixed GC中,G1垃圾收集器都会对每个Region的存活对象数量进行统计,根据存活对象数量和空闲Region的数量,动态地决定垃圾收集的区域和顺序,这种动态的垃圾收集策略,可以避免Full GC的发生,提高了应用程序的响应速度。

MixGC垃圾回收步骤

1)初始标记(STW):记录下GC Roots能直接引用的对象,并标记所有存活的对象,会执行一次年轻代GC,需要暂停所有线程,速度很快。

2)并发标记:与应用线程一起工作,进行可达性分析,G1收集器会对堆内存进行并发标记,找出所有存活的对象,并记录它们所在的Region.

3)最终标记(STW):修正并发标记期间, 部分因程序运行导致发生变化的那一部分对象,根据算法修复一些引用的状态;

4)筛选回收(STW):对各个Region的回收价值和成本进行排序,根据用户所期望的GC停顿STW时间,即-XX:MaxGCPauseMillis 制定计划。

这里举一个成本排序案例,现在有Region1、Region2和Region3三个区域:

        1)Region1预计可以回收1.5MB内存,预计耗时2MS,投产比ROI=1.5/2;

        2)Region2预计可以回收1MB内存,预计耗时1MS,投产比ROI=1/1;

        3)Region3预计可以回收0.5MB内存,预计耗时1MS,投产比ROI=0.5/1;

        那Region1、Region2和Region3各自的回收价值与成本比值分别是:0.75、1和0.5,比值越高说明同样的付出收益越高,如果此时只能回收一个Region的内存空间,G1就会选择Region2进行回收,保证了G1收集器在有限的时间内尽可能地提高收集效率。

常见参数讲解

        G1的使用方式非常的简单,只需对应开启G1垃圾收集器,配置堆最大内容及最大停顿时间便可享受到。一个搞笑的事情是在职场中,很大可能工龄十多年的老手花大半月时间去调优一次JVM参数,还不如直接升级JDK来的性能高,当然这也只是当成一个笑点,但这侧面也能看出来升级JDK的性价比,当然如果能掌握JDK,两者结合在一起加以利用能发挥它的最大效能。

相关参数

  • -XX:+UseG1GC:启用G1垃圾收集器。
  • -XX:G1HeapRegionSize=n:Java 堆大小划 分出约 2048 个区域,默认是堆内存的1/2000;配置需要为2的N次幂,1MB~32MB,使用G1垃圾回收器最小堆内存应为 1MB*2048=2GB ,低于这个的建议使用其它垃圾回收器。
  • -XX:MaxGCPauseMillis=n:设置最大停顿时间,单位为毫秒,默认为200毫秒(JVM会尽力实现,但不能保证达到)
  • -XX:ParallelGCThreads=n:设置 STW 工作线程数的值。一般设置为逻辑处理器的数量,最多为 8,是在STW阶段,并行执行【垃圾收集动作】的线程数
  • -XX:ConcGCThreads=n:在【并发标记】阶段,并发执行标记的线程数,一般将 n 设置为并行垃圾回收线程数 (ParallelGCThreads) 的 1/4
  • -XX:InitiatingHeapOccupancyPercent=n:设置G1 Mix垃圾回收的触发阈值,默认为45%

命令案例

//输入
-XX:+UseG1GC -XX:MaxGCPauseMillis=100 -Xms524m -Xmx524m -XX:+PrintCommandLineFlags 

//输出
-XX:G1ConcRefinementThreads=9 -XX:GCDrainStackTargetSize=64 -XX:InitialHeapSize=549453824 -XX:MaxGCPauseMillis=100 -XX:MaxHeapSize=549453824 -XX:+PrintCommandLineFlags -XX:ReservedCodeCacheSize=251658240 -XX:+SegmentedCodeCache -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:+UseG1GC 

使用场景

  • 大型应用程序:可以将堆内存划分为多个区域,以实现更加精细化的垃圾回收。
  • 高并发、低延迟:对响应时间要求较高的应用程序,如Web应用程序、电子商务等高并发场景。
  • 大内存应用:可以在垃圾回收过程中释放大量的空间,提高了内存的利用率。

配置注意事项

  • 不手工设置年轻代大小,比如使用 -Xmn 选项或 -XX:NewRatio 等设置年轻代大小

  • 暂停时间的目标不要太小,G1 的吞吐量目标是 90% 的应用程序时间和 10%的垃圾回收时间。如果把停顿时间调得非常低,如设置为10毫秒,很可能出现的结果就是由于停顿目标时间太短,导致每次回收内存只占堆内存很小的一部分, 收集器收集的速度跟不上分配器分配的速度, 导致垃圾慢慢堆积。应用运行时间一长就占满堆引发Full GC反而降低性能, 通常把期望停顿时间设置为一两百毫秒是比较合理的

  • 避免存活时间短的大对象:G1垃圾收集器对程序的代码质量要求较高,需要对程序的内存使用情况进行精细化管理,对应用程序的代码进行优化和调整

ZGC-新一代垃圾搜集器

什么是ZGC

        Z Garbage Collector 是Oracle公司开发一种可伸缩、低停顿时间的垃圾收集器,标记-复制算法(进行了改进)。垃圾回收过程几乎全部是并发,实际STW停顿时间极短,停顿时间控制10ms内,主要采用的染色指针和读屏障技术。

        在 JDK 11 中是实验性的特性引入,在 JDK 15 中 ZGC 可以正式投入生产使用,使用 –XX:+UseZGC 启用,ZGC 的堆内存也是基于 Region 来分布,和G1类似,但不再区分新生代老年代的,Region 支持动态地创建和销毁,大小不是固定。

有三种类型的 Region

  • 小型页面 Small Region:容量固定2MB,主要用于放置小于 256 KB 的小对象。
  • 中型页面Medium Region:容量固定32MB,主要用于放置大于等于 256 KB 小于 4 MB 的对象。
  • 大型页面Large Region:容量不固定 为N * 2MB, Region 是可以动态变化的,但必须是 2MB 的整数倍,最小支持 4 MB。

垃圾回收特点

  • 低停顿时间:ZGC最大的特点是在不增加延迟的情况下,能够处理非常大的内存数据,可以将停顿时间限制在10ms以内,对于需要快速响应的应用程序来说是非常重要;

  • 可伸缩性:可以处理非常大的内存数据,适应不同规模的应用程序,从小型应用程序到大型企业级应用程序;

  • 不需要分代:不需要将内存分为新生代和老年代,不需要复杂的内存回收算法;

  • 并发处理:采用了并发处理的方式来进行垃圾回收可以在应用程序运行的同时进行垃圾回收;

工作流程

  • 初始标记(STW):找 GC Roots 直接引用的对象,处理时间和GC Roots的数量成正比,停顿时间不随着堆的大小而增加;

  • 并发标记(没有STW):扫描剩余的所有对象,处理时间比较长,业务线程与GC线程同时运行,但这个阶段会有漏标问题;

  • 再标记(STW):通过算法解决漏标对象,和G1中的解决漏标的算法类似;

  • 并发转移准备(没有STW) :分析最有回收价值GC分页,即ROI计算;

  • 初始转移(STW):转移初始标记的存活对象和做对象重定位,时间和GC Roots的数量成正比,时间不随堆的大小而增加;

  • 并发转移(没有STW):对转移并发标记的存活对象做转移;

平台支持说明

        部分版本里面是实验性参数,需要加 -XX:+UnlockExperimentalVMOptions 才可以使用

平台是否支持支持版本
Linux/x64JDK 15 (Experimental since JDK 11)
Linux/AArch64JDK 15 (Experimental since JDK 13)
macOS/x64JDK 15 (Experimental since JDK 14)
Windows/x64JDK 15 (Experimental since JDK 14)
Windows/AArch64JDK 16
macOS/AArch64JDK 17
Linux/PowerPCJDK 18

验证参数(JDK17环境)  

参数: -XX:+UseZGC -XX:+PrintCommandLineFlags -Xms32m -Xmx32m

输出结果
-XX:InitialHeapSize=33554432 -XX:MaxHeapSize=33554432 -XX:MinHeapSize=33554432 -XX:+PrintCommandLineFlags -XX:ReservedCodeCacheSize=251658240 -XX:+SegmentedCodeCache -XX:+UseCompressedClassPointers -XX:-UseCompressedOops -XX:+UseZGC 

关于升级建议

        ZGC业界还没大规模使用,更多在实验性观望阶段,还存在变动和争议阶段,如果可能则预计26年~28年成为主流;当下我们开发的采用的垃圾收集器是G1收集器,23~25年会是主流。

        大家使用的话可以升级JDK,在JDK 11中首次支持ZGC,且JDK 11属于长期支持(Long Term Support,LTS)版本,Linux平台下JDK11开始支持ZGC,JDK14开始支持Mac和Windows。

        JVM调优中,升级JDK版本有较大收益,但是风险需要评估好。通常从兼容性和功能模块和性能风险进行评估。

  • 兼容性和功能模块:程序依赖很多三方jar包,有些jdk升级会把旧的API直接移除,导致项目直接编译不通过,版本升级,有些潜在的功能逻辑可能会被调整,导致些不可预测的故障产生,尤其是接锅侠身上;
  • 性能风险:短期测试相关数据是比较优,但是也需要能保持性能稳定。常规建议技术团队可以预发布环境进行稳定性测试,基于生产环境流量拷贝;

学习资料拓展:https://wiki.openjdk.org/display/zgc

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

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

相关文章

未来科技的前沿:深入探讨人工智能的进展、机器学习技术和未来趋势

文章目录 一、人工智能的定义和概述1. 人工智能的基本概念2. 人工智能的发展历史 二、技术深入:机器学习、深度学习和神经网络1. 机器学习2. 深度学习3. 神经网络 三、人工智能的主要目标和功能1. 自动化和效率提升2. 决策支持和风险管理3. 个性化服务和预测未来 本…

【目标检测】DEtection TRansformer (DETR)

一、前言 论文: End-to-End Object Detection with Transformers 作者: Facebook AI 代码: DEtection TRansformer (DETR) 特点: 无proposal(R-CNN系列)、无anchor(YOLO系列)、无NM…

本地大语言模型LLM的高效运行专家 | Ollama

Ollama简介 Ollama是一个开源的大型语言模型服务工具,它帮助用户快速在本地运行大模型。通过简单的安装指令,用户可以执行一条命令就在本地运行开源大型语言模型,如Llama 2。Ollama极大地简化了在Docker容器内部署和管理LLM的过程&#xff0…

ICode国际青少年编程竞赛- Python-1级训练场-基本操作

ICode国际青少年编程竞赛- Python-1级训练场-基本操作 1、 Dev.step(3)2、 Dev.step(1)3、 Dev.step(7)4、 Dev.step(-1)5、 Dev.step(-5)6、 Dev.step(3) Dev.step(-8)7、 Dev.turnRight() Dev.step(1)8、 Dev.turnLeft() Dev.step(1)9、 Dev.step(4) Dev.tur…

2024年Q1季度户外装备线上市场数据分析:垂钓类用品占据市场主流

五一期间,随着外出旅游、露营、游玩增多,消费者也将目光投向户外装备。 但今年Q1季度,由于季节变换、天气气候等原因,户外装备市场表现不如预期。根据鲸参谋数据显示,今年Q1季度,在线上电商平台&#xff0…

Java-异常处理-定义三角形类Triangle和异常三角形IllegalTriangleException类 (1/2)

任意一个三角形,其任意两边之和大于第三边。当三角形的三条边不满足前述条件时,就表示发生了异常,将这种异常情况定义为IllegalTriangleException类。 自定义异常类IllegalTriangleException: 当三角形的三条边不满足条件&#x…

数据结构学习/复习3--顺序表的实现/数组下标思想类编程题

一、顺序表的实现 realloc函数的原异地扩容 二、编程练习题 1. 2. 3.

解决WordPress无法强制转换https问题

原因:我在用cs的时候,突然老鸟校园网突然断了,客户端cs连不上了,进程也杀不死,cpu占用100%,只能重启,但是重启后我的blog网站打不开了 开始以为是Nginx的问题,重启它说配置出了问题…

STM32——GPIO篇

技术笔记! 1. 什么是GPIO? GPIO是通用输入输出端口(General-purpose input/output)的英文简写,是所有的微控制器必不可少的外设之一,可以由STM32直接驱动从而实现与外部设备通信、控制以及采集和捕获的功…

java中的字符串(String)常量池理解

下面创建String对象的方式一样吗? 上述程序创建对象类似,为什么s1和s2引用对象一样,但是s3和s4不一样呢? 在java程序中,许多基本类型的字面常量会经常用到,例如2,3.11,“hyy”等。为了提升程序…

算法学习系列(五十四):单源最短路的综合应用

目录 引言一、新年好二、通信线路三、道路与航线四、最优贸易 引言 关于这个单源最短路的综合应用,其实最短路问题最简单的就是模板了,这是一个基础,然后会与各种算法结合到一块,就是不再考察单个知识点了,而是各种知…

【RBAC鉴权】node-casbin基础教程

一、RBAC概述 RBAC鉴权,完整的英文描述是:Role-Based Access Control,中文意思是:基于角色(Role)的访问控制。这是一种广泛应用于计算机系统和网络安全领域的访问控制模型。 简单来说,就是通过…

a-table 控制列的展示和隐藏

一、业务场景: 最近在使用 Antd-vue 组件库的时候,a-table需要根据不同角色的权限显示和隐藏 columns的列 为了避免大家走弯路,为大家整理了一下,粘走可以直接用的那种 二、具体实现步骤: 1.在需要显示与隐藏的列增加一…

Android Studio实现简单的自定义钟表

项目目录 一、项目概述二、开发环境三、详细设计3.1、尺寸设置3.2、绘制表盘和指针3.3、动态效果 四、运行演示五、总结展望六、源码获取 一、项目概述 在安卓开发中,当系统自带的View已经无法满足项目需求时,就要自定义View。在Android中是没有与钟表有…

深度学习之基于Matlab BP神经网络烟叶成熟度分类

欢迎大家点赞、收藏、关注、评论啦 ,由于篇幅有限,只展示了部分核心代码。 文章目录 一项目简介 二、功能三、系统四. 总结 一项目简介 一、项目背景 烟叶的成熟度是评估烟叶品质的重要指标之一,它直接影响着烟叶的口感、香气和理化特性。传…

jupyter notebook切换conda虚拟环境

首先,切换到某个虚拟环境,本人切换到了d2l环境: (d2l) C:\Users\10129>pip install ipykernel然后,如代码所示安装ipykernel包 最后,按下述代码执行: (d2l) C:\Users\10129>python -m ipykernel i…

软件工程全过程性文档(软件全套文档整理)

软件项目相关全套精华资料包获取方式①:进主页。 获取方式②:本文末个人名片直接获取。 在软件开发的全过程中,文档是记录项目进展、决策、设计和测试结果的重要工具。以下是一个简要的软件全过程性文档梳理清单: 需求分析阶段…

Ex1-C6油气化工防爆轮式巡检机器人

Ex1系列防爆轮式巡检机器人整机采用防爆设计,防爆等级为Exd II CT4 Gb。机器人通过无轨3D形态导航技术,结合360度防爆云台和无线防爆充电桩,实现整套防爆标准,可广泛应用于石油、燃气、化工、冶金等II类爆炸环境中,代替…

【Qt QML】QLibrary加载共享库中的类

QLibrary是一个用于加载动态链接库(或称为共享库)的类。它提供了一种独立于平台的方式来访问库中的功能。 在QLibrary中,可以通过构造函数或setFileName()方法设置要加载的库文件名。当加载库文件时,QLibrary会搜索所有平台特定的…

VmWare 虚拟机没有网络解决办法

由于最近需要,装了个VM虚拟机,但是突然发现本机有网络,虚拟机却没有网络,更换了虚拟机的网络设置,都尝试过了 都不管用, 最后尝试了这种方法完美解决 还原网络默认设置 首先还原虚拟网络编辑器设置 启动V…