垃圾收集算法 r囧r小猫 2022-03-10 13:19 280阅读 0赞 ### 垃圾收集算法 ### * * 标记-清除算法 * 复制算法 * 标记-整理算法 * 分代收集算法 ## 标记-清除算法 ## 最基础的收集算法是“标记-清除”(Mark-Sweep)算法,算法分为“**标记**”和“**清除**”两个阶段:首先标记出所有需要回收的对象,在标记完成后统一回收掉所有被标记的对象。之所以说它是最基础的收集算法,是因为后续的收集算法都是基于这种思路并对其缺点进行改进而得到的。 它的主要缺点有两个: 1. 一个是效率问题,标记和清除过程的效率都不高; 2. 另外一个是空间问题,标记清除之后会产生大量不连续的内存碎片,空间碎片太多可能会导致,当程序在以后的运行过程中需要分配较大对象时无法找到足够的连续内存而不得不提前触发另一次垃圾收集动作。 标记-清除算法的执行过程如图所示。 ![在这里插入图片描述][watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl8zNjU4NjEyMA_size_16_color_FFFFFF_t_70] ## 复制算法 ## 为了解决效率问题,一种称为“复制”(Copying)的收集算法出现了\*\*,它将可用内存按容量划分为大小相等的两块,每次只使用其中的一块\*\*。**当这一块的内存用完了,就将还存活着的对象复制到另外一块上面,然后再把已使用过的内存空间一次清理掉**。这样使得每次都是对其中的一块进行内存回收,内存分配时也就不用考虑内存碎片等复杂情况,只要移动堆顶指针,按顺序分配内存即可,实现简单,运行高效。只是这种算法的代价是**将内存缩小为原来的一半,未免太高了一点**。复制算法的执行过程如图3-3所示。 ![在这里插入图片描述][watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl8zNjU4NjEyMA_size_16_color_FFFFFF_t_70 1] 现在的商业虚拟机都采用这种收集算法来回收新生代,IBM的专门研究表明,新生代中的对象98%是朝生夕死的,所以并不需要按照1:1的比例来划分内存空间。 而是将内存**分为一块较大的`Eden`空间和两块较小的`Survivor`空间**,每次**使用`Eden`和其中的一块`Survivor`** 。当回收时,**将`Eden`和`Survivor`中还存活着的对象一次性地拷贝到另外一块`Survivor`空间上,最后清理掉`Eden`和刚才用过的`Survivor`的空间**。HotSpot虚拟机默认`Eden`和`Survivor`的大小比例是8:1,也就是每次新生代中可用内存空间为整个新生代容量的90%(80%+10%),只有10%的内存是会被“浪费”的。**当`Survivor`空间不够用时,需要依赖其他内存(这里指老年代)进行分配担保(Handle Promotion)。** ## 标记-整理算法 ## “标记-整理”(Mark-Compact)算法,标记过程仍然与“标记-清除”算法一样,但后续步骤不是直接对可回收对象进行清理,**而是让所有存活的对象都向一端移动**,**然后直接清理掉端边界以外的内存**,“标记-整理”算法的示意图如图所示。 ![在这里插入图片描述][watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl8zNjU4NjEyMA_size_16_color_FFFFFF_t_70 2] ## 分代收集算法 ## 当前商业虚拟机的垃圾收集都采用“分代收集”(Generational Collection)算法,这种算法并没有什么新的思想,只是根据对象的存活周期的不同将内存划分为几块,根据情况合理使用前面的几个算法。 一般是把Java堆分为新生代和老年代,这样就可以根据各个年代的特点采用最适当的收集算法。 1. 在新生代中,每次垃圾收集时都发现有大批对象死去,只有少量存活,那就选用复制算法。 2. 而老年代中因为对象存活率高、没有额外空间对它进行分配担保,就必须使用“标记-清理”或“标记-整理”算法来进行回收。 注:本文参考自周志明的《深入理解Java虚拟机》,非常感谢大佬的书! [watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl8zNjU4NjEyMA_size_16_color_FFFFFF_t_70]: /images/20220310/42d303c71fa042a9a78188c3d2d16140.png [watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl8zNjU4NjEyMA_size_16_color_FFFFFF_t_70 1]: /images/20220310/810014806d4d44baae18c60d3e06fff5.png [watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl8zNjU4NjEyMA_size_16_color_FFFFFF_t_70 2]: /images/20220310/b5697dddd6ba47bba84b8fb93ba3ba28.png
相关 垃圾收集算法 一、新生代和老年代 新生代:主要存放新创建的对象,内存大小相对比较小,垃圾回收频繁。 老年代:存放生命周期比较长的对象,内存大小相对比较大,垃圾回收也没有那么频。 两 灰太狼/ 2023年10月01日 21:12/ 0 赞/ 3 阅读
相关 垃圾收集算法 标记-清除算法 Mark-Sweep 首先标记出所有需要回收的对象,在标记完成后统一回收所有被标记的对象。 由两个不足: 1. 效率问题 今天药忘吃喽~/ 2023年07月04日 04:52/ 0 赞/ 19 阅读
相关 垃圾收集算法 对象已死吗 引用计数法 > 给对象中添加一个引用计数器,每当一个地方引用它时,计数器值就加1;当引用失效时,计数器值就减1;任何时刻计数器为0的对象就是不可能再被使 喜欢ヅ旅行/ 2023年06月03日 12:59/ 0 赞/ 30 阅读
相关 垃圾收集算法 1.标记-清除算法 标记处所有需要回收的对象,标记完成之后统一回收所有被标记的对象 不足之处:效率问题,标记和清除两个过程的效率都不高,空间问题:会产生很多内存碎片,以后在 ╰半橙微兮°/ 2023年03月12日 09:19/ 0 赞/ 22 阅读
相关 垃圾收集算法 垃圾收集器与内存分配策略参考目录: > 1.[判断Java 对象实例是否死亡][Java] > 2.[ Java 中的四种引用][Java 1] > 3.[垃圾收集算 秒速五厘米/ 2022年06月04日 05:24/ 0 赞/ 234 阅读
相关 垃圾收集算法 由于垃圾算法的实现涉及大量程序细节,而且各个平台的虚拟机操作内存的方法又各不相同,因此我们不需要过多关注算法的实现。只需要了解算法的思想和发展 1、标记-删除算法 ![ 深碍√TFBOYSˉ_/ 2022年05月14日 14:23/ 0 赞/ 238 阅读
相关 垃圾收集算法 垃圾收集算法 标记-清除算法 复制算法 标记-整理算法 分代收集算法 标记-清除算法 最基础的收集算法是“标 r囧r小猫/ 2022年03月10日 13:19/ 0 赞/ 281 阅读
相关 垃圾收集算法 标记-清除算法 其他算法的基础 主要问题: 1、效率低 2、造成大量内存碎片 复制算法 为了解决标记-清除算法的缺陷,将内存开成两个相等区域,对象动态 ╰+哭是因爲堅強的太久メ/ 2022年02月01日 07:45/ 0 赞/ 278 阅读
相关 垃圾收集算法 垃圾收集算法 标记-清除算法 复制算法 标记-整理算法 分代收集算法 标记-清除算法 最基础的收集算法是“标记-清除“算法,算法分为“标记 - 日理万妓/ 2021年09月24日 01:20/ 0 赞/ 331 阅读
相关 垃圾收集算法 由于垃圾收集算法的实现涉及大量的程序细节,而且各个平台的虚拟机操作内存的方法又各不相同,因此本节不打算过多地讨论算法的实现,只是介绍几种算法的思想及其发展过程。 小灰灰/ 2020年05月14日 16:09/ 0 赞/ 807 阅读
还没有评论,来说两句吧...