什么是GC 标记-压缩算法

需要对标记清除和GC复制算法有一定了解

GC标记-压缩算法是由标记阶段压缩阶段构成。

标记阶段和标记清除的标记阶段完全一样。之后我们要通过搜索数次堆来进行压缩。

Lisp2 算法的对象

Donald E.Knuth

对象结构如图示:

Lisp2 算法在对象头中为forwarding指针留出空间,forwarding指针表示对象的目标地点。(设定forwarding时,还不存在移动完毕的对象)

概要

假设我们在下图所示的状态下执行GC

标记完后状态如下(过程与标记清除算法相同)

压缩完后的状态如下(可以看到,他们现在的位置是挨着的。)

这种算法并不会对对象的顺序产生影响,知识缩小了他们之间的空隙,让他们聚集在堆的一端。

步骤

压缩阶段代码

  1. compaction_phase(){
  2. set_forwarding_ptr() // 设置forwarding指针
  3. adjust_ptr() // 更新指针
  4. move_obj() // 移动对象
  5. }

步骤一:设定forwarding指针

首先程序会搜索整个堆,给活动的对象设定forwarding指针。初始状态下forwarding是NULL。

  1. set_fowarding_ptr(){
  2. scan = new_address = $heap_start
  3. while(scan < $heap_end)
  4. if(scan.mark = TRUE)
  5. scan.forwarding = new_address
  6. new_address += scan.size
  7. scan += scan.size
  8. }
  • scan 用来搜索堆中的指针,new_address指向目标地点的指针。
  • 一旦scan找到活动对象,forwarding指针就要被更新。按着new_address对象的长度移动。
  • 如下图示:

步骤二:更新指针

  1. adjust_ptr(){
  2. for(r :$roots) // 更新根对象的指针
  3. *r = (*r).forwarding
  4. scan = $heap_start
  5. while(scan < $heap_end)
  6. if(scan.mark == TRUE)
  7. for(child :children(scan)) // 通过scan 更新其他对象指针
  8. *child = (*child).forwarding
  9. scan += scan.size
  10. }
  • 首先更新根的指针
  • 然后重写所有活动对象的指针(对堆进行第二次的搜索)

步骤三:移动对象

搜索整个堆(第三次搜索),再将对象移动到forwarding指针的引用处。

  1. move_obj(){
  2. scan = $free = $heap_start
  3. while(scan < $heap_end)
  4. if(scan.mark == TRUE) // 判断是否是活动对象
  5. new_address = scan.forwarding // 获取对象要移动的地点
  6. copy_data(new_address, scan, scan.size) // 复制对象(移动对象)
  7. new_address.forwarding = NULL // 将forwarding改为NULL
  8. new_address.mark = FALSE // mark改为FALSE
  9. $free += new_address.size // 指针后移
  10. scan += scan.size // 指针后移
  11. }

  • 算法不会对对象本身的顺序进行改变,只会把对象集中在堆的一端。
  • 算法没有去删除对象,知识吧对象的mark设置为FALSE
  • 之后把forwarding改为NULL,标志位改为FALSE,将$free移动obj.size个长度。

优缺点

优点:可有效利用堆

使用整个堆在进行垃圾回收,没啥说的。任何的算法都是有得有失,用时间换空间。或者用空间换时间。重要的是它在这里

适不适用。

缺点:压缩花费计算成本

Lisp2 算法中,对堆进行了3次搜索。在搜索时间与堆大小成正相关的状态下,三次搜索花费的时间是很恐怖。也就是说,它的吞吐量要低于其他算法。时间成本至少是标记清除的三倍(当然不包含mutator)

Mark Compact GC (Part one: Lisp2)的更多相关文章

  1. Mark Compact GC (Part two :Two-Finger)

    目录 Two-Finger算法 前提 概要 步骤一:移动对象 步骤二:更新指针 优缺点 表格算法 概要 步骤一:移动对象群 和 构筑间隙表格 移动对象群 构筑间隙表格 步骤二:更新指针 优缺点 Two ...

  2. Mark Sweep GC

    目录 标记清除算法 标记阶段 深度优先于广度优先 清除阶段 分配 First-fit.Best-fit.Worst-fit三种分配策略 合并 优点 实现简单 与保守式GC算法兼容 缺点 碎片化 分配速 ...

  3. 1. GC标记-清除算法(Mark Sweep GC)

    世界上第一个GC算法,由 JohnMcCarthy 在1960年发布. 标记-清除算法由标记阶段和清除阶段构成. 标记阶段就是把所有的活动对象都做上标记的阶段. 标记阶段就是"遍历对象并标记 ...

  4. Gson序列化问题导致的内存溢出,tip:Background sticky concurrent mark sweep GC freed

    问题原因,如果在json model里面放了非可序列化的对象就会导致这中问题,可序列化的就是那些基础数据类型和集合类型,如果在里面放个Android的Activity或者adapter这类类型字段,变 ...

  5. copy GC 和 mark & compaction GC的算法异同

    先标记 然后 copy GC是,对所有child,判断, 如果child没有被访问过,那么拷贝到新地址,child的forwording指向新地址,child标记为已访问,把自己对child的引用改为 ...

  6. Java中9种常见的CMS GC问题分析与解决

    1. 写在前面 | 本文主要针对 Hotspot VM 中"CMS + ParNew"组合的一些使用场景进行总结.重点通过部分源码对根因进行分析以及对排查方法进行总结,排查过程会省 ...

  7. GC学习笔记

    GC学习笔记 这是我公司同事的GC学习笔记,写得蛮详细的,由浅入深,循序渐进,让人一看就懂,特转到这里. 一.GC特性以及各种GC的选择 1.垃圾回收器的特性 2.对垃圾回收器的选择 2.1 连续 V ...

  8. 非常详细GC学习笔记

    转载:http://blog.csdn.net/fenglibing/article/details/6321453 这是我公司同事的GC学习笔记,写得蛮详细的,由浅入深,循序渐进,让人一看就懂,特转 ...

  9. 很具体GC学习笔记

    GC学习笔记 这是我公司同事的GC学习笔记,写得蛮具体的,由浅入深,循序渐进,让人一看就懂,特转到这里. 一.GC特性以及各种GC的选择 1.垃圾回收器的特性 2.对垃圾回收器的选择 2.1 连续 V ...

随机推荐

  1. python 工具包安装

    (1)wxPython是python的常用gui yum install wxPython (2)numpy, scipy是常用的数学处理工具包 yum install scipy

  2. ios的notification机制是同步的还是异步的

    与javascript中的事件机制不同.ios里的事件广播机制是同步的,默认情况下.广播一个通知,会堵塞后面的代码: -(void) clicked { NSNotificationCenter *c ...

  3. Codeforces Round #271 (Div. 2) 解题报告

    题目地址:http://codeforces.com/contest/474 A题:Keyboard 模拟水题. 代码例如以下: #include <iostream> #include ...

  4. DNS SOA NS区别

    转自 http://bbs.51cto.com/thread-908637-1.html NS服务器里有两个比较重要的记录.一个叫SOA记录(起始授权机构) 一个叫NS(Name Server)记录( ...

  5. centos7 配置redis

    文件上传 yum -y install lrzsz 安装redis部署前操作 同时下载redis-.tar.gz安装包 yum -y install gcc-c++ yum -y install tc ...

  6. 30.algorithm排序小结

    如果容器中是类,如果要调用sort则需要重载操作符 "<" 包含头文件 #define _CRT_SECURE_NO_WARNINGS #include <vector ...

  7. HDFS的配额

  8. CMake入门之创建一个基于PCL的最小工程

    最近在学习PCL,借助Cmake可省去繁琐的添加包含目录和依赖库操作. 一个典型的CMakeLists.txt内容通常为: cmake_minimum_required(VERSION 2.6 FAT ...

  9. T_SQL 日期函数

    日期函数基数表达式的日期和时间或者是从时间间隔中返回值. GETDATE(),返回当前系统的日期和时间.例: SELECT GETDATE();  结果为:2010-05-18 15:53:08.92 ...

  10. ReactiveCocoa简单使用20例

    ReactiveCocoa简单使用20例 1. 观察值变化 你别动,你一动我就知道. //当self.value的值变化时调用Block,这是用KVO的机制,RAC封装了KVO @weakify(se ...