JVM常见的垃圾回收器

news2025/1/18 7:22:54

1、回收方法区:

        方法区回收价值很低,主要回收废弃的常量和无用的类。

方法区中的存储:
    方法区中存储的是加载的类的信息,常量,静态变量,即时编译后的代码等数据,所以回收的对象也就是这些内容。

何种场景需要回收:
    既然回收方法区不是必须的,虽然效率低下,但是当内存不够使用的时候依然是会抛出OOM的,那么我们需要知道什么场景下需要去回收方法区。首先我们需要弄明白方法区会回收那些对象。在JDK1.7之前,常量池是在方法区中的,在此版本及以后则将其移到堆中。基于目前版本主要是1.8及以上,故我们以1.8为准。在此版本上,主要回收的是无用的类。如何判定一个类是无用的:
 

  • 无用的类回收有以下条件:

        ①JAVA实例已经全部被回收,在堆中没有该类的示例存在;

        ②该类的classLoader被回收;

        ③该类的java.long.class对象没有被引用,不会被其他方法通过反射访问该类的方法

        与堆中的对象回收机制不同,不是不用即回收。HotSpot虚拟机提供了-Xnoclassgc参数进行控制是否回收。
 

2、垃圾回收器:

Serial、Serial Old、PawNew、CMS、Parallel Scavenge、Parallel Old、G1

  • Serial收集器,串行收集器是最古老,最稳定以及效率高的收集器,可能会产生较长的停顿,只使用一个线程去回收。
  • ParNew收集器,ParNew收集器其实就是Serial收集器的多线程版本。
  • Parallel收集器,Parallel Scavenge收集器类似ParNew收集器,Parallel收集器更关注系统的吞吐量。
  • Parallel Old 收集器,Parallel Old是Parallel Scavenge收集器的老年代版本,使用多线程和“标记-整理”算法
  • CMS收集器,CMS(Concurrent Mark Sweep)收集器是一种以获取最短回收停顿时间为目标的收集器。
  • G1收集器,G1 (Garbage-First)是一款面向服务器的垃圾收集器,主要针对配备多颗处理器及大容量内存的机器. 以极高概率满足GC停顿时间要求的同时,还具备高吞吐量性能特征

CMS垃圾收集器
 

CMS(并发标记清除收集器)(Concurrent Mark Sweep) 

        CMS是基于“标记-清除"算法实现的、以最短停顿时间为目标的垃圾收集器,CMS垃圾回收器主要针对老年代进行垃圾回收

        CMS回收垃圾的整个过程分为以下四个阶段:

                即初始标记阶段、并发标记阶段、重新标记阶段 和 并发清除阶段。

                (涉及STW,暂停用户线程的阶段主要是:初始标记 和 重新标记)

        CMS垃圾收集器在并发标记阶段并发清除阶段能让用户线程和GC线程并发执行,因此在整个垃圾收集过程中用户不会感到明显的卡顿:

  •         初始标记(Initial-Mark)阶段: 初始标记阶段会暂停所有的用户线程, 单线程标记与GC Roots直接关联的对象, 单线程标记是为了确保标记过程的准确性和一致性。这个阶段的主要任务仅仅只是标记出与GCRoots能直接关联到的对象。一旦标记完成之后就会立即进入并发标记阶段,恢复之前被暂停的所有用户线程。此外,由于与GC Roots直接关联对象比较少,所以初始标记阶段的速度非常快。
  •         并发标记(Concurrent-Mark)阶段:在并发标记阶段中,程序的工作线程会和垃圾回收线程并发执行或者交叉执行,垃圾回收线程会进行可达性分析,会从GC Roots直接关联的对象开始向下遍历堆中的对象图,标记所有可达的对象 。这个过程耗时较长但是不需要停顿用户线程。
  •         重新标记(Remark)阶段:由于工作线程在并发标记阶段可能会造成对象的引用发生变化,即原本可达的对象变成了垃圾对象,不可达对象变成了可达对象,所以重新标记阶段垃圾回收器会暂停所有工作线程,对在并发标记阶段中发生变化的对象进行重新标记、修正
    • 注意只能修正原有对象不能修正新增对象,即只能修正原有对象非可达变可达、可达变非可达。
  •         并发清除(Concurrent-Sweep)阶段:清理在标记阶段判断已经死亡的对象,释放内存空间。由于不需要移动存活对象,所以这个阶段也是可以与用户线程同时并发的

CMS缺点:

  1. 内存碎片问题:CMS 使用的是标记-清除算法,它会在回收垃圾对象之后产生大量的内存碎片。当老年代空间碎片太多时,如果无法找到一块足够大的连续内存存放对象时,将不得不提前触发一次Full GC。 

  2. 无法处理浮动垃圾:并发标记阶段用户线程可能会产生新对象,重新标记阶段只能修正原有对象非可达变可达、可达变非可达,不能新增,那么这些新增对象有可能在并发标记和并发清理阶段变为浮动垃圾对象,CMS 无法直接处理这些浮动垃圾,需要等到下一次 GC 才能回收。

  3. 并发标记引起的CPU资源紧张:在并发标记阶段,用户线程与垃圾回收线程并发执行,这可能会增加一定程度的 CPU 消耗

  4. 回收时要确保用户线程有足够内存:不能等老年代满了再回收,而是老年代内存到达某个阈值后回收,防止用户线程在并发执行过程中新创建对象导致内存不够,导致虚拟机补偿使用Serial Old收集器进行回收并处理内存碎片,从而浪费更多时间。默认情况下,当老年代使用了92%的空间后就会触发CMS垃圾回收机制。可以通过-XX:CMSInitiatingOccupancyFraction来设置 

  • 优点:
    • 并发速度快;
    • 低停顿:仅初始标记和重新标记阶段需要停顿用户线程,这两个阶段运行速度很快,CMS垃圾收集器在并发标记阶段并发清除阶段能让用户线程和GC线程并发执行,因此在整个垃圾收集过程中用户不会感到明显的卡顿。

演示非可达变为可达:

        在 main 方法中,首先创建了一个对象 obj,并将其设置为 null,使其成为不可达状态。然后,在后续代码中,通过 new 关键字重新创建了一个 MyClass 对象,并将其赋值给 obj,使得 obj 变为了可达状态。在这个过程中,就实现了非可达变可达的情况。

public class NonReachableToReachableExample {
    
    static class MyClass {
        // 类中的成员变量
        int data;
        
        // 类的构造函数
        MyClass(int data) {
            this.data = data;
        }
    }

    public static void main(String[] args) {
        // 创建一个 MyClass 对象并赋值给变量 obj,初始时它是不可达的
        MyClass obj = null;
        
        // 在并发标记阶段后,某个用户线程重新获取了对 obj 的引用,使得 obj 变为可达
        obj = new MyClass(10);
        
        // 在重新标记阶段,obj 被标记为可达
        // 在此之前,obj 是不可达的,但由于用户线程的活动,它变为了可达
        // 因此,在重新标记阶段需要修正这个状态
    }
}

G1垃圾回收器:

        G1(Garbage-First,垃圾优先收集器):

  •         G1垃圾回收器不再把堆划分为连续的分代,而是将堆内存分割成2048个大小相等的Region,各Region根据需要扮演伊甸园区、幸存区、老年代区、巨大区Humongous Region。垃圾优先收集器跟踪各Region里垃圾的回收价值(回收空间大小和预计回收时长),在后台维护一个优先级列表,每次根据用户设定允许的收集停顿时间,回收优先级最高的那些Region,以达到垃圾优先的效果。
  • 设置最大停顿时间:-XX:MaxGCPauseMillis=默认0.2s
  •          G1从整体来看是基于标记-整理算法实现的回收器,但从局部(两个Region之间)上看又是基于复制算法实现的。
  •         G1在JDK9之后成为了默认的垃圾回收器,取代了Parallel Scavenge 、Parallel Old默认组合,而CMS被声明为不推荐使用的垃圾回收器。  

步骤:

  •         初始标记:初始标记阶段会暂停所有的用户线程, 单线程标记与GC Roots直接关联的对象, 单线程标记是为了确保标记过程的准确性和一致性。这个阶段的主要任务仅仅只是标记出与GCRoots能直接关联到的对象。
  •         并发标记:在并发标记阶段中,程序的工作线程会和垃圾回收线程并发执行或者交叉执行,垃圾回收线程会进行可达性分析,从GC Roots直接关联的对象开始向下遍历堆中的对象图,标记所有可达的对象 。这个过程耗时较长但是不需要停顿用户线程。
  •         最终标记:重新标记所有存活的对象和上个阶段用户线程新产生的可达对象。并发停顿。采用SATB算法,效率比CMS重新标记高。并发停顿。
  •         筛选回收:根据优先级列表,回收价值高的一些Region,将存活对象通过标记复制算法复制到同类型的空闲Region。根据指定的最大停顿时间回收,因此可能来不及回收所有垃圾对象,但能保证回收到最高回收价值的垃圾。并发停顿。

初始标记阶段会暂停所有的用户线程, 单线程标记与GC Roots直接关联的对象, 单线程标记是为了确保标记过程的准确性和一致性。这个阶段的主要任务仅仅只是标记出与GCRoots能直接关联到的对象。一旦标记完成之后就会立即进入并发标记阶段,恢复之前被暂停的所有用户线程。此外,由于与GC Roots直接关联对象比较少,所以初始标记阶段的速度非常快。

Minor GC与Full GC分别在什么时候发生?

        新生代内存不够用时候发生MGC也叫YGC,JVM内存不够的时候发生FGC

引用的分类:

        强引用:GC时不会被回收

        软引用:描述有用但不是必须的对象,在发生内存溢出异常之前被回收

        弱引用:描述有用但不是必须的对象,在下一次GC时被回收

        虚引用(幽灵引用/幻影引用):无法通过虚引用获得对象,用PhantomReference实现虚引用,虚引用用来在GC时返回一个通知。
 

 你知道哪些JVM性能调优:

设定堆内存大小

-Xmx:堆内存最大限制。

设定新生代大小。 新生代不宜太小,否则会有大量对象涌入老年代

-XX:NewSize:新生代大小

-XX:NewRatio 新生代和老生代占比

-XX:SurvivorRatio:伊甸园空间和幸存者空间的占比

设定垃圾回收器 年轻代用 -XX:+UseParNewGC 年老代用-XX:+UseConcMarkSweepGC

JVM的永久代中会发生垃圾回收么?

        垃圾回收不会发生在永久代,如果永久代满了或者是超过了临界值,会触发完全垃圾回收(Full GC)。如果你仔细查看垃圾收集器的输出信息,就会发现永久代也是被回收的。这就是为什么正确的永久代大小对避免Full GC是非常重要的原因。请参考下Java8:从永久代到元数据区 (注:Java8中已经移除了永久代,新加了一个叫做元数据区的native内存区)

类的生命周期:
     

   类的生命周期包括这几个部分,加载、连接、初始化、使用和卸载,其中前三部是类的加载的过程,如下图:

加载,查找并加载类的二进制数据,在Java堆中也创建一个java.lang.Class类的对象

连接,连接又包含三块内容:验证、准备、初始化。 1)验证,文件格式、元数据、字节码、符号引用验证; 2)准备,为类的静态变量分配内存,并将其初始化为默认值; 3)解析,把类中的符号引用转换为直接引用

初始化,为类的静态变量赋予正确的初始值

使用,new出对象程序中使用

卸载,执行垃圾回收
 

 Java对象创建过程

1.JVM遇到一条新建对象的指令时首先去检查这个指令的参数是否能在常量池中定义到一个类的符号引用。然后加载这个类(类加载过程在后边讲)

2.为对象分配内存。一种办法“指针碰撞”、一种办法“空闲列表”,最终常用的办法“本地线程缓冲分配(TLAB)”

3.将除对象头外的对象内存空间初始化为0

4.对对象头进行必要设置
 

对象分配规则:

        对象优先分配在Eden区,如果Eden区没有足够的空间时,虚拟机执行一次Minor GC。

        大对象直接进入老年代(大对象是指需要大量连续内存空间的对象)。这样做的目的是避免在Eden区和两个Survivor区之间发生大量的内存拷贝(新生代采用复制算法收集内存)。

        长期存活的对象进入老年代。虚拟机为每个对象定义了一个年龄计数器,如果对象经过了1次Minor GC那么对象会进入Survivor区,之后每经过一次Minor GC那么对象的年龄加1,知道达到阀值对象进入老年区。

        动态判断对象的年龄。如果Survivor区中相同年龄的所有对象大小的总和大于Survivor空间的一半,年龄大于或等于该年龄的对象可以直接进入老年代。

        空间分配担保。每次进行Minor GC时,JVM会计算Survivor区移至老年区的对象的平均大小,如果这个值大于老年区的剩余值大小则进行一次Full GC,如果小于检查HandlePromotionFailure设置,如果true则只进行Monitor GC,如果false则进行Full GC。
 

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

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

相关文章

Qt笔记-解决子控制大小获取不正确(width和height)需要重制窗体后,才能获得正确的值

在Qt中,子控件的宽度和高度在构造后并不准确,而只有在调整窗口大小后才正确,这可能是因为子控件的布局或者约束尚未完全计算和应用。 为了解决这个问题,可以使用QTimer来延迟获取子控件的宽度和高度,以确保在布局和约…

Footprint Analytics 与 GalaChain 达成战略合作

​ Footprint Analytics 宣布与 GalaChain 达成战略合作。GalaChain 是 Gala 旗下的 Layer 1 区块链。此次合作标志着双方在游戏(包括 Gala Games) 、娱乐和金融等多个行业的区块链生态系统革新方面迈出了重要的一步。 GalaChain 致力于满足企业级项目的广泛需求&…

【网安小白成长之路】8.sql注入操作

🐮博主syst1m 带你 acquire knowledge! ✨博客首页——syst1m的博客💘 🔞 《网安小白成长之路(我要变成大佬😎!!)》真实小白学习历程,手把手带你一起从入门到入狱🚭 &…

基于ssm微信小程序的4S店客户管理系统

采用技术 基于ssm微信小程序的4S店客户管理系统的设计与实现~ 开发语言:Java 数据库:MySQL 技术:SpringMVCMyBatis 工具:IDEA/Ecilpse、Navicat、Maven 页面展示效果 管理员端 管理员登录 管理员首页 用户管理 门店管理 …

Java技术学习|消息队列|初级RabbitMQ

学习材料声明 黑马RabbitMQ快速入门教程,快速掌握rabbitmq、springAMQP消息中间件 是非常初级的学习,听说后续的高级课程会涉及到微服务之类的,所以等学完微服务再回来学。还有redis的高级部分也涉及了微服务,所以也都暂时停止学…

Pick-a-Pic:An open dataset of user preferences for text-to-image generation

1.introduction 1.创建了一个 数据集,每个示例包括一个提示,两个生成的图像以及一个指向首选图像的标签,或者在没有一个图像明显优于另一个时标记为平局。由真实用户创建,包括50w个示例。 2.利用真实用户偏好,训练一个评分函数,使用人类偏好数据和类似于instructGPT奖励…

我花了2元,15分钟,零基础入门了Llama3的微调。

Llama3在最新测评上已经可以接近闭源的GPT4,我们很快就可以拥有属于自己的“GPT4”了!于是,我昨天第一时间体验了Llama3的微调,成本不高,大概花了2元。 为了方便大家体验微调,我把需要的系统环境打包成了社…

递归、搜索与回溯算法:综合练习

例题一 解法: 算法思路: ⾸先,我们在第⼀⾏放置第⼀个皇后,然后遍历棋盘的第⼆⾏,在可⾏的位置放置第⼆个皇后,然后再遍历第三⾏,在可⾏的位置放置第三个皇后,以此类推&#xff0c…

鸿蒙OpenHarmony【集成三方SDK】 (基于Hi3861开发板)

OpenHarmony致力于打造一套更加开放完善的IoT生态系统,为此OpenHarmony规划了一组目录,用于将各厂商的SDK集成到OpenHarmony中。本文档基于Hi3861开发板,向平台开发者介绍将SDK集成到OpenHarmony的方法。 规划目录结构 三方SDK通常由静态库…

2024年4月最新注册香港苹果账号(Apple ID)并解决支付的教程

大陆的Apple ID仅仅能下载国内的一些APP,其实海外也有非常之多好用又好玩的APP需要大家来挖掘!发现这些海外优质APP就得需要一个海外苹果账号。这就是我今天为什么要写这篇文章的初衷! 注册香港Apple ID教程 1、首先到http://appleid.apple.com里注册一个国内的Ap…

ROS2 命令行工具---常用命令整理

本文主要介绍 ROS2 机器人操作系统的一些常用命令行工具及其使用方法,使用这些命令可以使机器人编程和调试变得更加简便。 在实际应用过程中,我们会经常用到命令行操作来辅助调试,更进一步的可以使用GUI工具辅助调试。 一、创建工作空间 跟…

数据结构——第7章 查找

1 线性表的查找 数据元素和顺序表的定义 typedef struct{KeyType key;InfoType otherinfo; }ElemType; typedef struct{ElemType *R;int length; }SSTable; 1.1 顺序查找 int Search_Seq(SSTable ST,KeyType key){ST.R[0].keykey;for(int iST.length;ST.R[i].key!key;i--);…

SQLAlchemy的使用

SQLAlchemy中filter函数的使用 https://blog.csdn.net/m0_67093160/article/details/133318889 创建临时字段 select id , CONCAT(‘内容’) AS fullname from example_table; Pandas数据类型转换_pandas转换数据类型 https://blog.csdn.net/qq_41404557/article/details/125…

用wps自带工具给图片做标注

在wps中,选中wps中的图片,右键选择【编辑】进入图片编辑器,在选项卡面板右侧选择【标注】工具,再选择【添加文本】工具,即可直接在图片上输入文字,标注完成后选择【覆盖原图】就完成标注任务。

【3200字干货】2024跨境电商5大市场:选品风向深度剖析

以下是针对马来西亚、新加坡、泰国、菲律宾和台湾这5个东南亚跨境电商市场的选品市场分析: 一、马来西亚 市场特点:马来西亚是东南亚第三大经济体,拥有年轻的消费群体和对跨境购物的偏好。网购消费力强,易上手爆单,跨…

tableau基础学习——添加标靶图、甘特图、瀑布图

标靶图 添加参考线 添加参考分布 甘特图 创建新的字段 如设置延迟天数****计划交货日期-实际交货日期 为正代表提前交货,负则代表延迟交货 步骤:创建——计算新字段 把延迟天数放在颜色、大小里面就可以 瀑布图 两个表按照地区连接 先做个条形图&…

Python构建学生信息管理系统:网站路由补充和首次运行

在之前的内容中,我们已经完成了学生信息管理系统(Student Information Management System, SIMS)的需求分析、环境搭建、数据库创建、项目结构的初始化,以及运行。正常做下来的朋友,会发现项目运行后输入http://127.0.…

vscode在json文件中添加注释

1.在设置中输入关联文件,点击添加项; 2.

ClickHouse 24.3 版本发布说明

本文字数:10774;估计阅读时间:27 分钟 审校:庄晓东(魏庄) 本文在公众号【ClickHouseInc】首发 北半球迎来春天,也是 ClickHouse 发布新版本的时候了。 发布概要 本次ClickHouse 24.3版本包含了1…

Android 系统充电动画

效果 Android获取电池充电状态是否为快充可参考. Android_source/frameworks/base/packages/SystemUI/src/com/android/systemui/statusbar/phone/StatusBar.java private int lastBatteryStatus;private final BroadcastReceiver mBatteryChangedReceiver new BroadcastRece…