当前位置: 首页 > news >正文

垃圾收集算法

分代收集理论

分代收集理论,建立在两个分代假说之上。

弱分代假说:绝大多数对象都是朝圣夕灭的。

强分代假说:熬过越多次垃圾收集的过程的对象就越难以消亡。

这两个分代假说奠定了垃圾收集器的一致设计原则:收集器应该将Java堆划分出不同的区域,然后回收对象依据年龄分配到不同的区域之中存储。一般的虚拟机都会分为新生代和老年代两个区域。新生代中每次垃圾回收都会有大量对象死去,而存活的对象会慢慢放到老年代中去。

由于新生代和老年代的存储对象的特点,我们可以认为在新生代中进行垃圾收集是十分具有性价比的,因为每次垃圾收集都能释放大量的内存空间;相比较而言老年的垃圾收集效率就不那么可观。因此可以进行分代垃圾收集,但是这时候又出现了新的问题,新生代中的对象有可能是被老年引用的,为了找出这部分对象,除了在GCRoots之外,还需要对老年代的对象进行扫描,需要遍历整个老年代,这无疑带来了巨大的内存负担。因此这就引出了第三个假说:跨代引用相对于同代引用仅占极少数。因此我们可以在新生代上建立一个全局的数据结构(记忆集),来代替老年代的全局扫描。这个结构把老年代分为若干小块,标识出老年代的哪一块内存会存在跨代引用。在此后的MinorGC时,只有包含了跨代引用的小块内存里的对象会加入GCRoots进行扫描。

MinorGC/YoungGC:新生代收集,指目标只是新生代的垃圾收集

MajorGC/OldGC:老年代收集,目标只是老年代的垃圾收集

FullGC:收集整个Java堆和方法区的垃圾收集

标记-清除算法:清除标记的对象

步骤:标记出所有需要回收的对象,标记完成后,统一回收掉所有被标记的对象。或者标记出需要存活的对象,清除未标记的对象。

缺点:

  1. 执行效率不稳定,如果Java堆中包含大量对象,而且其中大部分都是需要回收的,这时候必须进行大量标记和清除的动作,导致标记和清除两个过程的执行效率都随对象数量增长而降低。
  2. 内存空间的碎片化问题:标记,清除之后会产生大量不连续的内存碎片,空间碎片太多可能会导致以后程序在分配较大对象时无法找到足够的连续内存而触发另一次垃圾收集动作。

标记-复制算法:复制存活的对象到一端,其余的清除(新生代)

为了解决标记-清除算法面对大量可回收对象时执行效率低的问题,提出了半区复制的垃圾回收算法,它将可用内存按容量划分为大小相等的两块,每次只使用其中的一块。当这块内存用完了,就将还存活着的对象复制到另外一块上面,然后再把已使用过的内存空间一次性清理掉。如果内存中多数对象都是存活的,这种算法会产生大量的内存间复制的开销。

缺点:内存空间利用率不高,空间浪费太多。

现在的虚拟机采用这种方法去回收新生代,有研究专门对新生代“朝生夕灭”的特点做了更量化的诠释。新生代中有98%的对象熬不过第一轮垃圾收集,因此不需要按照一半一半的比例来划分新生代的内存空间。

针对“朝生夕灭”特点对象,提出了一种更优化的复制分代策略,称为“Appel式”回收,Serial,ParNew等新生代收集器都采用了这种模式。具体做法是,将新生代分为一块较大的Eden空间和两块较小的Survivor空间。每次分配内存只使用Eden和其中一块Survivor。发生垃圾收集时,将Eden和Survivor中仍然存活的对象一次性复制到另一块Survivor空间上,然后直接清理掉Eden和已用过的那块Survivor空间。HotSpot虚拟机默认Eden和Survivor的大小比例是8:1,这也就意味着有10%的内存被浪费,Appel式回收还有一个罕见情况的“逃生门”设计,当每次MinorGC中Survivor内存不足以容纳存活的对象时,就需要依赖其他内存区域(实际上是老年代)进行分配担保。

标记-整理算法:存活的对象移动到一端(移动式,老年代)

标记-整理的标记过程和标记-清除算法一样,区别就是不是直接对标记对象进行回收处理,而是让所有存活对象都向内存空间一端移动,然后直接清理掉边界以外的内存。

标记--清除和标记-整理的本质区别是是否是移动式的回收算法,前者是非移动的,后者是移动式的。

移动对象并更新所有引用这些对象将会是一项非常负重的操作,而且这种对象移动操作必须全程暂停用户应用程序(Stop The World)才能进行。

是否移动对象都存在弊端,移动则内存回收时更复杂(STW频率小),不移动会导致内存分配是更复杂。HotSpot虚拟机里关注吞吐量Parallel Old收集器是基于标记-整理算法,关注延迟的CMS收集器是基于标记-清除算法的,带来的内存碎片问题会使用一次标记-整理算法解决。

http://www.lryc.cn/news/518652.html

相关文章:

  • SQL-leetcode-262. 行程和用户
  • 太原理工大学软件设计与体系结构 --javaEE
  • Leetcode 139. 单词拆分 动态规划
  • python异常机制
  • 运行爬虫时可能遇到哪些常见问题?
  • BGP与CN2的区别 详解两者在网络传输中的应用与优势
  • Spring 项目 基于 Tomcat容器进行部署
  • “负载均衡”出站的功能、原理与场景案例
  • 02-51单片机数码管与矩阵键盘
  • 不同方式获取音频时长 - python 实现
  • 【python A* pygame 格式化 自定义起点、终点、障碍】
  • 12_Redis发布订阅
  • 归并排序:数据排序的高效之道
  • 【redis初阶】浅谈分布式系统
  • CatLog的使用
  • 头歌python实验:网络安全应用实践-恶意流量检测
  • 大模型WebUI:Gradio全解11——Chatbots:融合大模型的多模态聊天机器人(2)
  • 如何用 Python 实现简单的 AI 模型?
  • 单片机-直流电机实验
  • python【数据结构】
  • 详解Sonar与Jenkins 的集成使用!
  • 《笔记》青蛙跳台阶——斐波那契数列
  • SpringBoot3动态切换数据源
  • OSPF - 特殊区域
  • Linux 系统下磁盘相关指令:df、du、fdisk、lsblk
  • 基于单片机的肺功能MVV简单测算
  • 如何用Python编程实现自动整理XML发票文件
  • 腾讯云AI代码助手编程挑战赛-百事一点通
  • Spring学习笔记1
  • LeetCode 2185. Counting Words With a Given Prefix