IDisposable 接口

托管资源和非托管资源

  • 托管资源

    • CLR 控制和管理的内存资源,如程序中在 Heap 上分配的对象、作用域内的变量等;
    • GC 机制实现自动内存管理和托管堆的全权管理;
  • 非托管资源
    • CLR 不能控制管理的部分,如文件流Stream/数据库连接coonection/窗口句柄/组件COM等;
    • Finalize 方法(析构函数) GC 隐式自动调用,Dispose 方法手动强制显式调用;
    • 尽量避免使用 Finalize() 方法清理资源,推荐实现 Dispose() 方法供显式调用;

注:MSDN - 实现 Finalize() 方法或析构函数对性能可能会有负面影响。用 Finalize() 方法回收对象占用的内存至少需要两次垃圾回收,第一次调用析构函数,第二次删除对象。 GC 机制在回收托管对象内存之前,会先调用对象的析构函数。   

析构函数(Finalize方法) .vs. Dispose方法

Finalize 方法用于释放非托管资源,Dispose 方法用于清理或释放由类占用的非托管和托管资源。IDisposable 接口定义见上,自定义类应实现 IDisposable 接口,设计原则:

  • 可以重复调用 Dispose() 方法;
  • 析构函数应该调用 Dispose() 方法;
  • Dispose() 方法应该调用 GC.SuppressFinalize() 方法,指示垃圾回收器不再重复回收该对象;

在一个包含非托管资源的类中,资源清理和释放的标准模式是:

  1. 继承 IDisposable 接口;
  2. 实现 Dispose() 方法,在其中释放托管和非托管资源,并将对象从垃圾回收器链表中移除;
  3. 实现类的析构函数,在其中释放非托管资源;

其中,变量 "isDisposing" 来区分手动显式调用(true)还是GC隐式调用(false)。

  public class MyDispose : IDisposable
{
public MyDispose() { }
~MyDispose() {
Dispose(false);
} private bool isDisposed = false;
public void Dispose(){
Dispose(true);
System.GC.SuppressFinalize(this);
}
protected virtual void Dispose(bool isDisposing) // 子类可重写
{
if (false == this.isDisposed)
{
if (true == isDisposing){
OtherManagedObject.Dispose(); // 释放托管资源 ...
}
OtherUnManagedObjectDisposeOrClose(); // 释放非托管资源 ...
this.isDisposed = true;
}
}
}

析构函数执行在类的实例被销毁之前需要的清理或释放非托管资源的行为,注意不能在析构函数中释放托管资源。类的析构函数被编译后自动生成 protected void Finalize() 方法,GC 垃圾回收时会调用该方法并对继承链中的所有实例递归地调用 Finalize() 方法。

Object.Finalize() 方法不可重写。

  • 类的析构函数不可继承和重载、不能带访问修饰符,一个类至多有一个析构函数;
  • 析构函数只针对类的实例对象,没有静态析构函数;
 protected void Finalize(){
try{
//
}
finally{
base.Finalize();
}
}

Finalize() 方法被调用的情况:

  • 显式调用System.GC 的 Collect方法(不建议);
  • Windows 内存不足、第G0代对象充满;
  • 应用程序被关闭或 CLR 被关闭;

Dispose() 方法的调用分 2 种:

  • 使用 using 语句会自动调用:using( MyDispose myObj = new MyDispose() ) {…}
  • 显式调用:myObj.Dispose();

一个资源安全的类,都应实现 IDisposable 接口和析构函数,提供手动释放资源和系统自动释放资源的双保险。(1)若一个类A有一个实现了 IDisposable 接口类型的成员并创建(创建而不是接收,必须是由类A创建)它的实例对象,则类A也应该实现 IDisposable 接口并在 Dispose 方法中调用所有实现了 IDisposable 接口的成员的 Dispose 方法;(2)如果基类实现了 IDisposable 接口,那么其派生类也要实现 IDisposable 接口,并在其 Dispose 方法中调用基类中 Dispose 方法;只有这样才能保证所有实现了 IDisposable 接口的类的对象的 Dispose 方法能被调用到、手动释放任何需要释放的资源。

参考

为什么 IEnumerator 接口没有继承 IDisposable 接口
托管资源和非托管资源IDisposable接口的一个典型例子
Finalize - Dispose - SuppressFinalizeIDisposable和Finalize的区别和联系
对.Net 垃圾回收 Finalize 和 Dispose 的理解
深刻理解 C# 中资源释放

GC 垃圾回收

本质:跟踪所有被引用到的对象,整理不再被引用的对象并回收相应内存。

优点

  • 减少由于内存运用不当产生的Bug,降低编程复杂度;
  • 高效的内存管理;
  • 提高软件系统的内聚;

代 Generation

NET 垃圾回收器将 CLR 托管堆内的对象分为三代:G0、G1、G2,代龄机制支持有选择地查询,提高垃圾回收性能,避免回收整个托管堆。

  • G0:小对象(Size<85000Byte),最近被分配内存的对象,支持快速存取对象;
  • G1:在GC中幸存下来的G0对象,CLR 检查过一次未被回收的G0对象;
  • G2:大对象(Size>=85000Byte),CLR 检查过二次及以上仍未被回收的G1/G2对象;

通过 GC.GetGeneration() 方法可以返回对象所处的代。当第0代对象已满时,自动进行垃圾回收,第0代中未被释放的对象成为第1代,新创建的对象成为第0代,以此类推,当第0代再次充满时会再次执行垃圾回收,未被释放的对象被添加到第1代。随着程序的执行,第1代对象会产生垃圾,此时垃圾回收器并不会立即执行回收操作,而是等第1代被充满回收并整理内存,第1代中未被释放的对象成为第2代。当第1代收集时,第0代也需要收集,当第2代收集时,第1和第0代也需要收集。

根 root

每个应用程序都包含一组根,每个根都是一个存储位置,包含一个指针或引用托管堆上的一个对象或为null,由 JIT编译器 和 CLR运行时 维护根(指针)列表。

工作原理

基于代的垃圾回收器如下假设:

  • 对象越新,生存期越短,最近分配内存空间的对象最有可能被释放,搜索最近分配的对象集合有助于花费最少的代价来尽可能多地释放内存空间;
  • 对象越老,生存期越长,被释放的可能性越小,经过几轮GC后,对象仍然存在,搜索代价大、释放内存空间小;
  • 程序的局部性原理 :同时分配的内存对象通常同时使用,将它们彼此相连有助于提高缓存性能和回收效率;
  • 回收堆的一部分速度快于回收整个堆;

标记和清除 (Mark & Sweep) 收集算法:避免出现 "环引用" 造成内存泄露
利用内部结构的 终止队列(Finalization Queue) 跟踪保存具有 Finalize 方法(定义了析构函数)的对象。

  • ReRegisterForFinalize():将对象的指针重新添加到Finalization队列中;(允许系统执行Finalize方法)
  • SuppressFinalize():将对象的指针从Finalization 队列中移除;(拒绝系统执行Finalize方法)

程序创建具有 Finalize 方法的对象时,垃圾回收器会在终止队列中添加一个指向该对象的项(引用或指针)。当对象不可达时,没有定义析构函数的不可达对象直接由 GC 回收,定义了析构函数的不可达对象从终止队列中移除到 终止化-可达队列(F-reachable Queue)中。在一个特殊的专用线程上,垃圾回收器会依次调用该队列中对象的 Finalize 方法并将其从队列中移除,执行后该对象和没有Finalize方法的垃圾对象一样,然后在下一次 GC 中被回收。(GC线程 和 Finalizer线程 不同)
算法分 2 步:

  • 标记阶段:垃圾识别。从应用程序的 root 出发,利用相互引用关系,递归标记(DFS),存活对象被标记,维护一张树图:"根-对象可达图"; 
  • 压缩阶段:内存回收。利用 Compact 压缩算法,移动内存中的存活对象(大对象除外)并修改根中的指针,使内存连续、解决内存碎片问题,有利于提高内存再次分配的速度和高速缓存的性能;  

参考

C#基础知识梳理系列十一:垃圾回收机制步步为营 C# 技术漫谈 四、垃圾回收机制(GC)
垃圾回收机制 - Generation的原理分析
详解 Finalization队列与 F-reachable队列深入浅出理解 GC 机制
垃圾回收GC:.Net自动内存管理系列

内存泄漏

按照编译原理,内存分配策略有3种:

  • 静态存储区(方法区):编译时即分配好,程序整个运行期间都存在,主要存放静态数据、全局static数据和常量
  • 栈区:局部变量,自动释放
  • 堆区:malloc或new的动态分配区,需手动释放

推荐使用 .Net 内存分析工具:CLR Profiler,用来观察托管堆内存分配和研究垃圾回收行为的一种工具。

附注:

该处提供一个狂降内存的方法(摘自网上),可以极大优化程序内存占用。
这个函数是将程序的物理内存尽可能转换为虚拟内存,大大增加硬盘读写,是不好的,慎用!!
使用方法:在程序中用一个计时器,每隔几秒钟调用一次该函数,打开任务管理器

    [DllImport("kernel32.dll", EntryPoint = "SetProcessWorkingSetSize")]
public static extern int SetProcessWorkingSetSize(IntPtr process, int minSize, int maxSize);
/// <summary>
/// 释放内存
/// </summary>
public static void ClearMemory()
{
GC.Collect();
GC.WaitForPendingFinalizers();
if (Environment.OSVersion.Platform == PlatformID.Win32NT)
{
SetProcessWorkingSetSize(System.Diagnostics.Process.GetCurrentProcess().Handle, -1, -1);
}
}

  

C# ~ 由 IDisposable 到 GC的更多相关文章

  1. C# ~ 从 IEnumerable / IEnumerator 到 IEnumerable<T> / IEnumerator<T> 到 yield

    IEnumerable / IEnumerator 首先,IEnumerable / IEnumerator 接口定义如下: public interface IEnumerable /// 可枚举接 ...

  2. C#GC垃圾回收和析构函数和IDisposable的使用

    一,什么是GC 1,GC是垃圾回收器,一般来说系统会自动检测不会使用的对象或变量进行内存的释放,不需要手动调用,用Collect()就是强制进行垃圾回收,使内存得到及时的释放,让程序效率更高. 2,G ...

  3. C#技术漫谈之垃圾回收机制(GC)

    GC的前世与今生 虽然本文是以.NET作为目标来讲述GC,但是GC的概念并非才诞生不久.早在1958年,由鼎鼎大名的图林奖得主John McCarthy所实现的Lisp语言就已经提供了GC的功能,这是 ...

  4. 关于GC和析构函数的一个趣题

    这个有趣的问题感谢装配脑袋友情提供. 请看如下代码: public class Dummy { public static Dummy Instance; ; ~Dummy() { Instance ...

  5. C#中对IDisposable接口的理解

    http://blog.sina.com.cn/s/blog_8abeac5b01019u19.html C#中对IDisposable接口的理解 本人最近接触一个项目,在这个项目里面看到很多类实现了 ...

  6. C# 中正确实现 IDisposable 接口

    作用 此接口的主要用途是释放非托管资源. 当不再使用托管对象时,垃圾回收器会自动释放分配给该对象的内存. 但无法预测进行垃圾回收的时间. 另外,垃圾回收器对窗口句柄或打开的文件和流等非托管资源一无所知 ...

  7. 从C#垃圾回收(GC)机制中挖掘性能优化方案

    GC,Garbage Collect,中文意思就是垃圾回收,指的是系统中的内存的分配和回收管理.其对系统性能的影响是不可小觑的.今天就来说一下关于GC优化的东西,这里并不着重说概念和理论,主要说一些实 ...

  8. .NET面试题解析(06)-GC与内存管理

      系列文章目录地址: .NET面试题解析(00)-开篇来谈谈面试 & 系列文章索引 GC作为.NET的重要核心基础,是必须要了解的.本文主要侧重于GC内存管理中的一些关键点,如要要全面深入了 ...

  9. C# IDisposable接口

    public class MyClass : IDisposable { public int a; public MyClass() { //构造 } public void Dispose() { ...

随机推荐

  1. 用批处理文件进行TCP/IP设置,方便在家与办公IP切换

    在公司用公司分配的固定IP上网,回家后又要将本本设置为家里的固定IP上网,每次都要手动重复一个过程: 打开网络中心,选择本地连接,进入属性然后选择IPV4进行TCP/IP的设置,填入IP,子网掩码DN ...

  2. [开源]微信在线信息模拟测试工具(基于Senparc.Weixin.MP开发)

    目前为止似乎还没有看到过Web版的普通消息测试工具(除了官方针对高级接口的),现有的一些桌面版的几个测试工具也都是使用XML直接请求,非常不友好,我们来尝试做一个“面向对象”操作的测试工具. 测试工具 ...

  3. IOS Animation-KeyPath值

    IOS Animation-KeyPath值 keyPath值 说明 值类型 position 移动位置 CGPoint opacity 透明度 0-1 bounds 变大与位置 CGRect bou ...

  4. sql的OUTER APPLY

    今天写一个查询sql,其中涉及到一个银行卡绑定表(表名:BankBind),我要将这个表的开户行字段的值进行分割出省份.支行, 这个开户行字段存储的值如“广东省广东省分行江门市分行恩平市支行”.“招商 ...

  5. MVVM架构~knockoutjs与MVC配合,实现列表的增删改功能

    返回目录 MVC与MVVM的模型 在MVC实例项目中,为我们提供了简单的增删改查功能,而这种功能的实现与具体的Model很有关系,或者说它与后台数据库的关系过于紧密了,而对于开发人员来说当页面布局修改 ...

  6. DataGrid--多记录CRUD

    最近在做一个datagrid,但因为引用的Jquery,加上初学者,所以难免费尽周折.现在将完整版贴出来,跟大家分享,一起切磋,也方便自己回顾学习. ps:第一次发帖,不知排版效果如何,瑕疵勿怪. 首 ...

  7. PC端和移动端地址适配

    判断当前页面的打开方式是pc还是移动设备,如果是移动设备,跳转到对应移到端网站的方法: 方法一.还是用@media screen 思路:css使用媒体查询,当屏幕小于760px时,使某个元素的样式发生 ...

  8. ng-表单验证

    表单验证<AngularJs>   常用的表单验证指令 1. 必填项验证 某个表单输入是否已填写,只要在输入字段元素上添加HTML5标记required即可: <input type ...

  9. Geometry Curve of OpenCascade BRep

    Geometry Curve of OpenCascade BRep eryar@163.com 摘要Abstract:几何曲线是参数表示的曲线 ,在边界表示中其数据存在于BRep_TEdge中,BR ...

  10. jxl读取excel实现导入excel写入数据库

    @RequestMapping(params = "method=import", method = RequestMethod.POST) @ResponseBody publi ...