`
hqs7636
  • 浏览: 221018 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

垃圾回收 2.014

阅读更多
无版本差异

D 是一种全面采用垃圾回收(Garbage Collection)的语言。这意味着它从来不用释放内存。只需要按需分配,然后由垃圾回收程序周期性的将所有未使用的内存返回给可用内存池。

C 和 C++ 程序员习惯于显式的管理内存分配和释放,很可能会怀疑垃圾回收的好处和功
效。对一开始就采用垃圾回收设计的新项目和用垃圾回收改良的现有项目的经验表明:

• 采用垃圾回收的程序更快。这有些违反直觉,但是其原理有:
• 引用计数是解决显式内存分配问题的常用解决方案。实现赋值时递增和递减操

作的代码通常是程序缓慢的原因之一。将其隐藏在智能指针类之后并不能提高
速度。 (无论如何,引用计数也不是全面的解决方案,因为循环引用从不会
被删除。)

• 析构函数用来释放对象获得的资源。对于大多数类来说,这个资源就是内存。
采用垃圾回收,大多数析构函数都可以完全抛弃。

• 当对象在堆栈上分配时,那些负责释放内存的析构函数就变得十分重要。对于
这些函数来说,必须确立一种机制以使异常发生时,每一个函数帧中的析构函
数都会被调用以释放对象持有的内存。如果析构函数变得不相关,就没有设置
特殊的堆栈帧用于处理异常,这样运行也会更快。

• 内存管理代码总量会变少。程序越大,它在cache中的部分的越少;它的调页
越多,它运行的越慢。

• 垃圾回收只会在内存变得紧张时才会运行。当内存尚且宽裕时,程序将全速运
行,不会在释放内存上花费任何时间。

• 相对于过去的缓慢的垃圾回收程序,现代的垃圾回收程序要先进得多。分代,
复制回收程序在很大程度上克服了早期的标记&清除算法的低效。

• 现代垃圾回收程序进行堆紧缩。堆紧缩将减少程序引用的页的数量,这意味着
内存访问命中率将更高,交换将更少。

• 采用垃圾回收的程序不会因为内存泄漏的累积而崩溃。

• 垃圾回收程序回收不被使用的内存,因此不会有“内存泄露”问题;内存泄露会使长
期运行的应用程序逐渐耗尽内存直至使系统崩溃。采用 GC 的程序拥有更长期的稳定
性。

• 采用垃圾回收的程序 有更少的难以发现的指针错漏。 这是因为没有指向已经释放的
内存的悬挂指针。因为没有显式的内存管理代码,也就不可能有相应的错漏。

• 采用垃圾回收的程序的开发和调试更快,因为不用开发、调试、测试或维护显式的释
放代码。

• 采用垃圾回收的程序会明显更小,因为没用用于管理内存释放的代码,也不需要处理
内存释放异常的代码。


垃圾回收并非什么仙丹妙药。它有着以下不足:
• 内存回收何时运行是不可预测的,所以程序可能意外暂停。
• 运行内存回收的时间是没有上界的。尽管在实践中它的运行通常很快,但无法保证这
一点。
• 除了回收程序以外的所有线程在回收进行时都会停止运行。
• 垃圾回收程序也许会留下一些本该回收的内存。在实践中,这不是什么大问题,因为
显式内存回收程序通常会泄露一些内存,这致使它们最终耗尽所有内存,另一个理由
就是显式内存回收程序通常会把内存放回自己的内部内存池中而不是把内存交还给操
作系统。
• 垃圾回收应该被实现为一个基本的操作系统内核服务。但是因为现实并非如此,就造
成了采用垃圾回收的程序被迫带着它们的垃圾回收实现到处跑。尽管这个实现可以被
做成一个共享 DLL ,它也还是程序的一部分。
这些限制可以通过采用 内存管理 中介绍的技术来缓解。


22.1 垃圾回收如何工作

GC 的工作方式:
1. 寻找所有在 GC 所分配的内存里的指针“根”。
2. 递归扫描所有分配的被“根”所指向的内存,查找更多在GC 分配内存内的指针
3. 释放所有没有激活指针指向的 GC 分配的内存。
4. 可能的话就通过复制分配的对象(叫做复制回收器)压缩余下被占用的内存。


22.2 外部代码同垃圾回收对象的协作

垃圾回收程序寻找根的地方有:
1. 它的静态数据段
2. 每个线程的堆栈和寄存器
3. 所有由 std.gc.addRoot() 或 std.gc.addRange() 添加的根
如果对象唯一的根不在它们中,回收程序将会释放它占有的内存。
要避免这种行为,则需要:
• 为对象维护一个根,该根要位于回收程序扫描的区域之内。
• 添加一个根到使用 std.gc.addRoot() 或 std.gc.addRange() 对象里。
• 使用外部代码的存储分配程序重新分配一个对象或者使用 C 运行时库的 malloc/free



22.3 指针和垃圾回收程序

D 中的指针可以被粗略地分为两类:指向垃圾回收内存的指针和其他的指针。后者包括由调
用 C 的 malloc() 所创建的指针,从 C 库中获得的指针,指向静态数据的指针,指向堆栈上
对象的指针,等等。对于这些指针,所有 C 中的合法指针的操作都可以应用其上。
但是,对于垃圾回收指针和引用,对指针的操作就有一些限制了。这些限制是不会造成太大
的缺陷,但却会使垃圾回收程序的设计灵活许多。
未定义行为(Undefined behavior):
• 将指针同其他的值进行异或,如同在 C 中在链表中用异或保存指针那种小技巧。
• 不要使用异或技巧交换两个指针的值。
• 使用类型转换和其他的小技巧将指针存储在非指针变量内。
void* p;
...
int x = cast(int)p; // 错误:未定义行为

垃圾回收器在构建根集(root)时不会扫描非指针类型。
• 利用特定的指针对齐的特点在低位或者高位存储 flag :
p = cast(void*)(cast(int)p | 1); // 错误:未定义行为
• 将可能指向垃圾回收堆的值转换为指针:
p = cast(void*)12345678; // error:未定义行为


复制垃圾回收器可能会更改这个值。
• 不要将不同于 null 的“魔数”存入指针。
• 不要将指针值写入到磁盘然后再从磁盘读入到内存。
• 不要使用指针值来计算 hash(哈希)函数。采用复制技术的垃圾回收程序可能会任
意地在内存中移动对象,这会使 hash 值失效。
• 不要依赖指针的顺序:

if (p1 < p2) // 错误:未定义行为
...

因此,再一次提醒,垃圾回收程序可能会在内存中移动对象。

• 不要给指针加上或者减去一个偏移量,不然结果有可能会导致指针指向垃圾回收程序
原来为对象分配的范围之外。

char* p = new char[10];
char* q = p + 6; // 正确
q = p + 11; // 错误:未定义行为
q = p - 1; // 错误:未定义行为

• 如果这些指针可能指向 GC 堆的话,请不要位移它们,如:
align (1) struct Foo
{ byte b;
char* p; // 位移(misaligned)指针
}

如果底层硬件支持位移指针,而且该指针从未用来指向该 GC 堆,则也可以使用它
们。

• 不要使用“一个字节一个字节的”内存复制方式来复制指针值。这样可能会导致中间
条件,到时就没有合法的指针了;而且如果 GC 在此种条件下暂停线程的话,那么它
就会搞乱内存。大部分的 memcpy() 实现是可以工作的,因为它的内部实现完成的
是以排列块大于或等于一个指针大小的方式进行的复制,不过由于 C 标准并不担保
此种实现,因此在使用 memcpy() 时应特别小心。
可靠且可以完成的事情:

• 使用联合共享指针的存储空间:
union U { void* ptr; int value }

• 如果存在指向垃圾回收对象对象内部的指针,就不必维护指向对象开始处的指针。
char[] p = new char[10];
char[] q = p[3..6];
// q 完全可以用来指向整个对象,不需要同时
// 保留 p 。

程序员可以不用指针完成大部分任务。D 提供的各种特征使得人们在决大多数情况下都不需
要显式地使用指针,这些特征包括:引用对象、动态数组和垃圾回收。提供指针的目的是成
功地同 C API 衔接并完成一些低级的工作。


22.4 与垃圾回收程序协作

垃圾回收程序并不能解决所有的内存释放问题。例如,如果保留了一个指向一大块数据的指
针,那么就算它再也用不到了,垃圾回收程序也无法回收这块空间。为了解决这个问题,一
个不错的实际操作就是:在某个对象不再使用时将指向它的引用或指针设置为 null。
这个建议只适用于静态引用或者嵌入到其他对象内的引用。它对于那些存储在堆栈上的引用
没什么意义,因为回收程序并不扫描栈顶之上的部分,并且新的堆栈帧总会被初始化。



22.5 参考
• Wikipedia
• GC 问与答
• 统一处理器——垃圾回收器技术
• 垃圾回收:自动的动态内存管理算法
分享到:
评论

相关推荐

    基于动态优先级的实时Java垃圾回收策略.pdf

    2. 垃圾回收器(GC):垃圾回收器是一种自动管理内存的机制,用于回收Java对象在堆中的内存空间。 3. 优先级:优先级是一种机制,用于确定线程或任务的优先级,以确保关键线程或任务的执行。 4. 实时Java:实时Java...

    24.java垃圾收集机制.zip

    24.java垃圾收集机制.zip24.java垃圾收集机制.zip24.java垃圾收集机制.zip24.java垃圾收集机制.zip24.java垃圾收集机制.zip24.java垃圾收集机制.zip24.java垃圾收集机制.zip24.java垃圾收集机制.zip24.java垃圾收集...

    毕设&课程作业_这是智能垃圾回收系统.zip

    2. **系统设计**:智能垃圾回收系统的设计需要考虑多个方面,如用户界面、后台管理、数据存储、算法实现等。用户界面应直观易用,后台管理应方便工作人员监控和调度,数据存储需确保数据安全,而算法则用于优化垃圾...

    建筑垃圾回收利用.pdf

    2. 建筑垃圾的产量与增长:建筑垃圾的产量巨大,每平方米新建建筑产生500吨至600吨,每平方米拆除建筑产生7000吨至1.3万吨。以某市为例,5年间日产量从1万吨增加到近3万吨,年产量接近1000万吨。 3. 建筑垃圾的传统...

    垃圾分类回收.doc

    2. 循环经济:循环经济是指在生产和消费过程中,通过减少、再利用和回收,使资源得到最大限度的利用,降低废弃物的产生,实现经济和环境的可持续发展。 3. 可持续发展:可持续发展是指在满足当前社会经济发展需求的...

    Java对象的清除与垃圾回收.pdf

    Java对象的清除与垃圾回收 Java中的垃圾回收机制是其主要特性之一,本文将介绍垃圾回收的概念和finalize()方法的使用。 一、垃圾回收的机制 Java虚拟机的垃圾回收机制是一种自动的内存管理机制。与C++不同,Java...

    南京科远DCS软件 NT6000 V4.2.014 Release Build 2021.1018 Setup

    南京科远DCS软件 NT6000 V4.2.014 Release Build 2021.1018 Setup

    HttpWatchPro-v6.014(网页数据分析工具)

    HttpWatch是一款强大的网页数据分析工具.集成在Internet Explorer工具栏.包括网页摘要.Cookies管理.缓存管理.消息头发送/接受.字符查询.POST 数据和目录管理功能.报告输出

    最新清理电脑垃圾脚本.bat

    最新清理电脑垃圾脚本.bat最新清理电脑垃圾脚本.bat最新清理电脑垃圾脚本.bat最新清理电脑垃圾脚本.bat最新清理电脑垃圾脚本.bat最新清理电脑垃圾脚本.bat最新清理电脑垃圾脚本.bat最新清理电脑垃圾脚本.bat最新清理...

    Java垃圾回收新算法刍探.pdf

    2. 垃圾回收算法:垃圾回收算法是垃圾回收器实现的核心,决定了垃圾回收的方式和效率。 本文提出了一种新的垃圾回收算法,该算法可以一定程度上改善和提高Java垃圾回收的性能。 Java与C/C++的对比: 1. 内存管理...

    垃圾分类表格.pdf

    1. 可回收垃圾:这类垃圾主要包括纸类、塑料、玻璃、金属和布料等。它们可以通过回收、再加工转化为新的产品。例如,废纸可以回收造纸,废弃塑料瓶可以再造为塑料制品,废旧衣物可以分解用于生产隔音材料或再生纤维...

    垃圾分类书.docx

    垃圾分类书.docx

    垃圾管理计划.pdf

    2. **加工垃圾程序**:确定操作设备和人员,以及不同类型的垃圾处理方法,包括再利用和回收。 3. **垃圾储存程序**:设立储存站,标明用途和容量,保持良好的存储条件以防止污染。 4. **排放程序**:遵循MARPOL附则V...

    垃圾回收分类系统、垃圾回收分类平台、垃圾回收分类、智慧管理系统、订单管理、财务管理、系统管理、库存管理、设备管理、Axure高保真web端后台管理系统.rp

    垃圾回收分类系统、垃圾回收高保真原型设计、垃圾分类后台管理系统、垃圾回收分类平台、垃圾回收分类、智慧管理系统、订单管理、财务管理、系统管理、库存管理、设备管理、Axure高保真web端后台管理系统 Axure原型...

    Java垃圾回收机制研究.pdf

    Java垃圾回收机制研究

    开题报告SpringBoot垃圾分类回收系统.docx

    毕业设计开题报告

    毕业设计论文SpringBoot垃圾分类回收系统.docx

    毕业设计论文

    文献综述(垃圾的可回收)英文版.doc

    2. 国内垃圾处理的传统方式主要是填埋、堆肥和焚烧。 3. 国外已经取得了不错的垃圾回收成果,例如瑞典已经实现了高达99%的垃圾回收率。 4. 我国居民的垃圾分类知识还较为简单和片面化。 5. 垃圾分类设施存在混乱的...

    建筑垃圾处理措施.pdf

    3. 废弃物管理:设置废弃物储存场地,分类存放,运输至指定地点处理,可回收废弃物应再回收利用。 六、其他管理 1. 对易燃、易爆物品及化学品实行专项管理,防止污染。 2. 保证施工机械良好运行,避免噪声和废弃物...

    EhLib 9.3 Build 9.3.014 Cracked

    Delphi 著名控件包 Ehlib,支持 RAD Studio 10.3 RIO。

Global site tag (gtag.js) - Google Analytics