垃圾回收 - 分代垃圾回收

news2024/9/20 16:44:47

分代垃圾回收在对象中导入了“年龄”的概念,通过优先回收容易成为垃圾的对象,提高垃圾回收的效率。

1、新生代对象和老年代对象

分代垃圾回收中把对象分类成几代,针对不同的代使用不同的 GC 算法,我们把刚生成的对象称为新生代对象,到达一定年龄的对象则称为老年代对象。
众所周知,新生代对象大部分会变成垃圾。如果我们只对这些新生代对象执行 CC会怎么样呢?除了引用计数法以外的基本算法,都会进行只寻找活动对象的操作(如标记清除算法的标记阶段和 复制算法等)。因此,如果很多对象都会死去,花费在 GC上的时间应该就能减少。
我们将对新对象执行的 CC 称为新生代 GC (minor GC)。mainor 在这里的意思是“小规模的”。新生代 GC 的前提是大部分新生代对象都没存活下来,GC 在短时间内就结束了。
另一方面,新生代 GC 将存活了一定次数的新生代对象当作老年代对象来处理。我们把类似于这样的新生代对象上升为老年代对象的情况称为晋升 (promotion).。
因为老年代对象很难成为垃圾,所以我们对老年代对象减少执行 GC 的频率。相对于新生代GC,我们将面向老年代对象的 GC 称为老年代 GC (major CC)。
在这里有一点需要注意,那就是分代垃圾回收不能单独用来执行 GC。我们需要把它和之前介绍的基本算法结合在一起使用,来提高那些基本算法的效率。
也就是说,分代垃圾回收不是跟GC 标记一清除算法和 GC 复制算法并列在一起供我们选择的算法,而是需要跟这些基本算法一并使用。

2、Unger的分代垃圾回收

在Ungar 的分代垃圾回收中,堆的结构如下图所示。我们总共需要利用 4个空间,分别是生成空间、2个大小相等的率存空间以及老年代空间,并分别用 new_start、survivor1_start、survivor2_start、old_start 这4 个变量引用它们的开头。我们将生成空间和幸存空间合称为新生代空间。新生代对象会被分配到新生代空间,老年代对象则会被分配到老年代空间里。Ungar 在论文里把生成空间、幸存空间以及老年代空间的大小分别设成了 140K字节、28K 字节和 940K宇节。
此外我们准备出一个和堆不同的数组,称为记录集(remembered set),设为rs。

在这里插入图片描述

生成空间就如它的字面意思一样,是生成对象的空间,也就是进行分配的空间。当生成空间满了的时候,新生代 GC 就会启动,将生成空间中的所有活动对象复制,这跟GC 复制算法是一个道理。目标空间是幸存空间。
2个幸存空间和CC复制算法里的 From 空间、To 空间很像,我们经常只利用其中的一个。在每次执行新生代 GC 的时候,活动对象就会被复制到另一个幸存空间里。在此我们将正在使用的幸存空间作为 From 幸存空间,将没有使用的幸存空间作为To幸存空间。
不过新生代GC 也必须复制生成空间里的对象。也就是说,生成空间和 From 幸存空间这两个空间里的活动对象都会被复制到To幸存空间里去。这就是新生代 CC。
只有从一定次数的新生代 CC 中存活下来的对象才会得到晋升,也就是会被复制到老年代空间去。
在这里插入图片描述

分代垃圾回收的优点是只将垃圾回收的重点放在新生代对象身上,以此来缩减GC所需要的时间。不过考虑到从老年代对象的引用,结果还是要搜索堆中的所有对象,这样一来就大大削减了分代垃圾回收的优势。所以就利用到了——记录集。
记录集用来记录从老年代到新生代对象的引用。这样新生代GC就可以不搜索老年代空间的所有对象,只通过搜索记录集来发现从老年代到新生代对象的引用。

3、记录集

记录集被用于高效地寻找从老年代对象到新生代对象的引用。具体来说,在新生代GC时将记录集看成根,并进行搜索,以发现指向新生代空间的指针。
在这里插入图片描述

记录集基本上是用固定大小的数组来实现的。各个元素是指向对象的指针。
那么,我们该怎么往记录集里记录对象呢?这就是下一节要介绍的“写入屏障”了。

4、写入屏障

在分代垃圾回收中,为了将老年代对象记录到记录集里,我们利用写入屏障 (write barrier)。在mutator 更新对象间的指针的操作中,写入屏障是不可或缺的。write_barrier()函数的伪代码如下所示。这个函数跟引用计数法中出现的 update_ptr()函数是在完全相同的情况下被调用的。

write_barrier (obj, field, new_obj){
	if (obj >= $old_start && new_obj < $old_start && obj.remembered == false)
		$rs [$rs-indez] = obj
		$rs_index++
		obj.remembered = true
	*field = new_obj
}

参数obj是发出引用的对象,obj 内存在要更新的指针,而 field指的就是 obj 内的域。new_obj 是在指针更新后成为引用目标的对象。
if判断主要是检查以下三点:

  • 发出引用的对象是不是老年代对象
  • 指针更新后的引用的目标对象是不是新生代对象
  • 发出引用的对象是否还没有被记录到记录集中

当这些检查结果都为真时,obj 就被记录到记录集中了。

5、对象的结构

在Ungar 的分代垃圾回收中,对象的头部中除了包含对象的头部中类和大小之外,还有以下这3条信息:

  • 对象的年龄(age)
  • 已经复制完成的标志(forwarded)
  • 已经向记录集记录完毕的标志(remembered)

age表示的是对象从新生代 CC 中存活下来的次数,这个值如果超过一定次数(AGE_MAX),对象就会被当成老年代对象处理。我们在 CC 复制算法和 GC标记压缩算法中也用到过 forwarded, 这里它的作用是一样的,都是用来防止重复复制相同对象的标志。这里的remembered也一样,是用来防止向记录集中重复记录的标志。不过 remembered只用于老年代对象,age 和forwarded只用于新生代对象。
此外,跟GC复制算法一样,在这里我们也使用forwarding指针。在torvrarting 指针中利用obj.field1,用obj.forwarding 访问 obj.field1.
Ungar 的分代垃圾回收中用到的对象的结构如下图所示
在这里插入图片描述

6、分配

分配是在生成空间进行的。执行分配的new_obj()函数如下所示

new_obj (size){
	if ($new_free + size ›= $survivor1_start)
		minor_gc ()
		if ($new_free + size ›= $survivor1_start)
			allocation_fail ()
 		}
	obj = $new_free
	$new_free += size
	obj.age = 0
	obj.forwarded = false
	obj.remembered = false
	obi.size = size
	return obj
}

7、优缺点

优点:吞吐量得到改善
缺点:在部分程序中会起到反作用。“很多对象年纪经轻就会死”这个法则毕竟只适合大多数情况,并不适用于所有程程序。当然、对象会活得很久的程序也有很多。对这样的程序执行分代垃圾回收,就会产生以下两个问题

  • 新生代GC所花费的时间增多
  • 老年代GC频繁运行
    而且,写入屏障导致的额外负担降低了吞吐量。只有当新生代GC 带来的速度提升效果大于写入屏障对速度造成的影响时,分代垃级回收才能够更好地发挥作用。当这个大小关系不成立时,分代址圾回收就没有什么作用,或者说反而可能会起到反作用。这种情况下我们还是使用基本算法更好。

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

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

相关文章

UI自动化测试详解

前言 随着智能化信息基础设施的推进&#xff0c;软件开发的进程也不断加快。软件测试工作也逐渐由传统的手工测试向软件自动化测试跨越。 对于很多企业来说&#xff0c;做好软件自动化测试工作已经不仅仅是通过测试工具进行“点点点”&#xff0c;要想找出软件测试过程中的缺…

python 小案例72

import requestsdef fetch_data_from_api(url):response requests.get(url)if response.status_code 200:data response.json()return dataelse:print("Failed to fetch data from API")return None# 使用NASA的API获取每日天文图片 url "https://api.nasa.…

【Springcloud】Actuator服务监控

【Springcloud】Actuator服务监控 【一】基本介绍【二】如何使用【三】端点分类【四】整合Admin-Ui【五】客户端配置【六】集成Nacos【七】登录认证【八】实时日志【九】动态日志【十】自定义通知 【一】基本介绍 &#xff08;1&#xff09;什么是服务监控 监视当前系统应用状…

情侣头像微信小程序源码 朋友圈背景小程序源码 动态壁纸微信小程序源码

壁纸和情侣头像&#xff0c;朋友圈素材都可以做&#xff0c;带视频教程。 搭建也不难&#xff0c;纯前端无后台。直接开发者工具调试前端&#xff0c;绑定合法域名&#xff0c;流量主功能也是在前端替换。 无需服务器域名直接上手&#xff01;&#xff01;&#xff01;

飞行动力学 - 第17节-part3-垂尾和推进系统对航向的影响 之 基础点摘要

飞行动力学 - 第17节-part3-垂尾和推进系统对航向的影响 之 基础点摘要 1. 尾翼的贡献2. 垂尾是航向静稳定性的最大来源3. 推进系统对航向的贡献3.1 螺旋桨3.2 喷气式 4. 参考资料 1. 尾翼的贡献 平尾对航向静稳定性的影响机理与机翼相同&#xff0c;由于尺寸小&#xff0c;通…

AI教程 | 用Midjourney制作AI模特和换装的保姆级教程

Hi! 大家好&#xff0c;我是专注于AI项目实战的赤辰。 昨天电商朋友过来交流&#xff0c;聊到他最近新开了一家淘宝店&#xff0c;在没有请任何员工的情况下&#xff0c;他一个人用AI工具完成了店铺取名&#xff0c;商品文案&#xff0c;店铺logo&#xff0c;主图设计&#xf…

ASO优化之阅读并回复应用的评论

回复评论对于与用户保持牢固的关系非常重要。如果时间有限&#xff0c;优先回复负面评论&#xff0c;可以向其他用户保证&#xff0c;我们正在积极解决应用的问题&#xff0c;从而提高转化率。 1、逻辑与沟通要清晰。 首先&#xff0c;无论他们的反馈是正面还是负面&#xff0…

【c++】如何有效地利用命名空间?

​ &#x1f331;博客主页&#xff1a;青竹雾色间 &#x1f618;博客制作不易欢迎各位&#x1f44d;点赞⭐收藏➕关注 ​✨人生如寄&#xff0c;多忧何为 ✨ 目录 前言什么是命名空间&#xff1f;命名空间的语法命名空间的使用避免命名冲突命名空间的嵌套总结 前言 当谈到C编…

51单片机-直流电机学习

简介 51单片机采用的是5V的直流电机 轴长&#xff1a;8mm 轴径&#xff1a;2mm 电压&#xff1a;1-6V 参考电流&#xff1a;0.35-0.4A 3V 转速&#xff1a;17000-18000 转每分钟 他的组成&#xff1a; 直流电机的结构应由 定子 和 转子 两大部分组成。 直流电机运行时静止…

【Spring】aop的底层原理

&#x1f384;欢迎来到边境矢梦的csdn博文&#x1f384; &#x1f384;本文主要梳理 Spring 中的切面编程aop的底层原理和重点注意的地方 &#x1f384; &#x1f308;我是边境矢梦&#xff0c;一个正在为秋招和算法竞赛做准备的学生&#x1f308; &#x1f386;喜欢的朋友可以…

工作和生活中,如何用项目管理思维解决复杂的事情?

在工作和生活中&#xff0c;许多事情都可以采用项目思维方式来解决。当我们逐渐将工作和生活中的各种事务以项目的方式来处理和推进时&#xff0c;我们可能并没有意识到&#xff0c;实际上我们正在运用项目管理思维。 项目管理思维能帮助我们在面对繁杂事务时&#xff0c;理清…

DevOps到底是什么意思?

前言: 当我们谈到 DevOps 时,可能讨论的是:流程和管理,运维和自动化,架构和服务,以及文化和组织等等概念。那么,到底什么是"DevOps"呢? 那么,DevOps是什么呢? 有人说它是一种方法,也有人说它是一种工具,还有人说它是一种思想。更有甚者,说它是一种哲学…

【echarts】如何修改折线图X轴每个刻度的间隔宽度,让拥挤的空间变大,所有坐标点的文案可以显示得下,Echarts x轴文本内容太长的几种解决方案

Echarts 如何修改折线图X轴每个刻度的间隔宽度&#xff0c;让拥挤的空间变大&#xff0c;所有坐标点的文案可以显示得下&#xff0c;Echarts x轴文本内容太长的几种解决方案 有以下几种方案&#xff0c;堪称最全方案&#xff1a; 1、dataZoom进行坐标的比例缩放 通过调整dataZ…

生态第五篇-调度的多维空间技术

生态第五篇-调度的多维空间技术 文章目录 生态第五篇-调度的多维空间技术前言一、什么是多维空间&#xff1f;二、实现原理1.先看效果2.如何实现 预告 前言 调度已经结束更新了本不想再更新调度技术&#xff0c;因为生态的更新计划里面有这一条所以就写一篇把 一、什么是多维…

Java“牵手”ebay商品详情数据,ebay商品详情API接口,ebayAPI接口申请指南

天猫平台商品详情接口是开放平台提供的一种API接口&#xff0c;通过调用API接口&#xff0c;开发者可以获取天猫商品的标题、价格、库存、月销量、总销量、库存、详情描述、图片等详细信息 。 获取商品详情接口API是一种用于获取电商平台上商品详情数据的接口&#xff0c;通过…

了解Armv8.x和Armv9.x扩展

概述 Arm架构新增的功能以扩展的形式提供&#xff0c;这样Arm能够定期发布新功能&#xff0c;以响应合作伙伴的需求&#xff0c;而无需对主架构进行重大更改。 Arm 每年都会发布新的扩展。Cortex CPU 是该架构的 Arm 实现&#xff0c;其会根据发布时间使用相应的扩展。 本指…

扫描mapper包

文章目录 第一种-配置在resource目录下第二种- 直接配置java代码目录&#xff0c;在Maven中配置相关路径 第一种-配置在resource目录下 第二种- 直接配置java代码目录&#xff0c;在Maven中配置相关路径 不配置不会把mapper的xml文件编译到target文件中 <build><res…

vs2022不能加载winform界面

今天遇到vs2022加载winform界面错误&#xff0c; The service ‘Microsoft.VisualStudio.Shell.Design.Serialization.DesignerDocDataService’ must be installed for this feature to work。 当前vs版本&#xff1a; Microsoft Visual Studio Enterprise 2022 (64 位) - Cur…

虚拟化和容器

文章目录 1 介绍1.1 简介1.2 虚拟化工作原理1.3 两大核心组件&#xff1a;QEMU、KVMQEMUKVM 1.4 发展历史1.5 虚拟化类型1.6 云计算与虚拟化1.7 HypervisorHypervisor分为两大类 1.8 虚拟化 VS 容器 2 虚拟化应用dockerdocker 与虚拟机的区别 K8Swine 参考 1 介绍 1.1 简介 虚…

【MySQL】初见数据库

目录 什么是MySQL 为什么要使用数据库 数据库基础 数据库的本质 存储引擎 常用操作 登录mysql 创建数据库 使用数据库 查看数据库 创建数据库表 查看表 向表中插入数据 查询表中数据 什么是MySQL &#x1f352;在我们服务器安装完 MySQL 服务之后&#xff0c;经…