终于开始动手写这篇文章了,有个网友催了我好几次,而我要么因为手头有事,要么就是被其他思路给叉开,以至这篇文章拖了好久还没开始写,今天终于可以静下心来完成它。

用了.net工具来写程序的人,不难发现它有个好处,就是使用的内存不用释放,尤其在使用C#或者VB.Net来写程序,因为程序所占用的内存都是受系统托管的,因此内存的释放不需要程序员去操心。

很多人从C语言或者C++等等语言转型过来,对于这一点往往很不适应,例如定义一个数组或者去new一个对象的时候,都习惯在使用完毕后用Delete语句去释放它,然而在C#中没有提供类似的语句来进行同样的操作。

那么有人就问,是不是.Net不用去释放内存,或者问假如要去显示释放一个对象,该如何去做。

那么我要明确告诉一点的就是,在C#中内存的回收是GC去做的,我们在程序中最多只是标记当前对象不再被引用就行了(而GC何时去回收是不确定的,因为回收内存是比较费时费力的,被触发的可能性在于内存紧张或者显示调用GC.Collect)。明白这一点后,那么我们在写程序的时候,当所定义的类型使用了比较大的内存资源或者使用了会引起操作冲突的资源,例如:各种连接对象,各种Stream对象,各种与图有关的对象,各种互斥对象等等,需要提供接口来进行关闭和标记,从而在GC回收的时候能提高效率。

.Net提供了三种方法,也是最常见的三种,大致如下:

1. 析构函数;

2. 继承IDisposable接口,实现Dispose方法;

3. 提供Close方法。

对于析构函数来说,长时间使用C#的人们,都会对它产生淡忘。或者说用C#编写一个类的时候,很少编写类的析构函数。而对于C#的析构函数来说,基本上延用了原来C++中的意思。但是在C#中不能像C++那样显示去删除一个对象,那么对象的析构函数调用是当GC检测到此对象不再被引用时,才进行删除,此时才会被调用。而对于GC何时去检测和收集是不确定的,因此对象的析构函数调用时机也是不确定的。这里也暗藏了一个道理,就是在析构函数中去做一些资源的关闭和标记就不是很合理了,因为所占有的资源无法迅速地进行关闭或者标记为无用。

析构函数不能显示调用,而对于后两种方法来说,都需要进行显示调用才能被执行。Close与Dispose这两种方法的区别在于,调用完了对象的Close方法后,此对象有可能被重新进行使用;而Dispose方法来说,此对象所占有的资源需要被标记为无用了,也就是此对象要被销毁,不能再被使用。例如常见.Net类库中的SqlConnection这个类,当调用完Close方法后,可以通过Open重新打开一个数据库连接,当彻底不用这个对象了就可以调用Dispose方法来标记此对象无用,等待GC回收。明白了这两种方法的意思后,大家在往自己的类中添加的接口时候,不要歪曲了这两者意思。

接下来说说这三个函数的调用时机,我用几个试验结果来进行说明,可能会使大家的印象更深。

首先是这三种方法的实现,大致如下:

/**//// 

    /// The class to show three disposal function

    /// 

    public class DisposeClass:IDisposable

    {

        public void Close()

        {

            Debug.WriteLine( "Close called!" );

        }

 

        ~DisposeClass()

        {

            Debug.WriteLine( "Destructor called!" );

        }

 

        IDisposable Members#region IDisposable Members

 

        public void Dispose()

        {

            // TODO:  Add DisposeClass.Dispose implementation

            Debug.WriteLine( "Dispose called!" );

        }

 

        #endregion

    }

对于Close来说不属于真正意义上的释放,除了注意它需要显示被调用外,我在此对它不多说了。而对于析构函数而言,不是在对象离开作用域后立刻被执行,只有在关闭进程或者调用GC.Collect方法的时候才被调用,参看如下的代码运行结果。

private void Create()

{

DisposeClass myClass = new DisposeClass();

}

private void CallGC()

{

GC.Collect();

}

// Show destructor

Create();

Debug.WriteLine( "After created!" );

CallGC();

运行的结果为:

After created!

Destructor called!

显然在出了Create函数外,myClass对象的析构函数没有被立刻调用,而是等显示调用GC.Collect才被调用。

对于Dispose来说,也需要显示的调用,但是对于继承了IDisposable的类型对象可以使用using这个关键字,这样对象的Dispose方法在出了using范围后会被自动调用。例如:

using( DisposeClass myClass = new DisposeClass() )

{

//other operation here

}

如上运行的结果如下:

Dispose called!

那么对于如上DisposeClass类型的Dispose实现来说,事实上并没有达到标记内存无用的目的,也就是说对象的析构函数还会被调用。

那么有人就问,既然Dispose方法中去为了显示标记此对象已经不再引用,那么调用对象的析构函数已经没有什么意义,是否能在Dispose中增加处理,来避免析构函数的调用。答案是有的,就是需要在Dispose方法中,加上调用GC.SuppressFinalize(this )的语句。那么改写后的DisposeClass如下:

 /**//// 

    /// The class to show three disposal function

    /// 

    public class DisposeClass:IDisposable

    {

        public void Close()

        {

            Debug.WriteLine( "Close called!" );

        }

 

        ~DisposeClass()

        {

            Debug.WriteLine( "Destructor called!" );

        }

 

        IDisposable Members#region IDisposable Members

 

        public void Dispose()

        {

            // TODO:  Add DisposeClass.Dispose implementation

            Debug.WriteLine( "Dispose called!" );

            GC.SuppressFinalize( this );

        }

 

        #endregion

    }

通过如下的代码进行测试。

private void Run()

{

using( DisposeClass myClass = new DisposeClass() )

{

//other operation here

}

}

private void CallGC()

{

GC.Collect();

}

// Show destructor

Run();

Debug.WriteLine( "After Run!" );

CallGC();

运行的结果如下:

Dispose called!

After Run!

显然对象的析构函数没有被调用。通过如上的实验以及文字说明,大家会得到如下的一个对比表格。

析构函数

Dispose方法

Close方法

意义

销毁对象

销毁对象

关闭对象资源

调用方式

不能被显示调用,在GC回收是被调用

需要显示调用

或者通过using语句

需要显示调用

调用时机

不确定

确定,在显示调用或者离开using程序块

确定,在显示调用时

那么在定义一个类型的时候,是否一定要给出这三个函数地实现呢。

我的建议大致如下。

1.  一般不要提供析构函数,因为它不能及时地被执行;

2.  对于Dispose和Close方法来说,需要看所定义的类型所使用的资源(参看前面所说),而决定是否去定义这两个函数;

3.  在实现Dispose方法的时候,一定要加上“GC.SuppressFinalize( this )”语句。

C#程序所使用的内存是受托管的,但不意味着滥用,好地编程习惯有利于提高代码的质量以及程序的运行效率。

转载于:https://www.cnblogs.com/yfcomeon/archive/2008/01/24/1051870.html

浅谈C#托管程序中的资源释放问题 (转载)相关推荐

  1. 浅谈线程池(中):独立线程池的作用及IO线程池

    在上一篇文章中,我们简单讨论了线程池的作用,以及CLR线程池的一些特性.不过关于线程池的基本概念还没有结束,这次我们再来补充一些必要的信息,有助于我们在程序中选择合适的使用方式. 独立线程池 上次我们 ...

  2. 嵌入式开发-浅谈嵌入式MCU开发中的三个常见误区

    浅谈嵌入式MCU开发中的三个常见误区 原创 2017-09-30 胡恩伟 汽车电子expert成长之路 目录 (1)嵌入式MCU与MPU的区分 (2)误区一:MCU的程序都是存储在片上Flash上,然 ...

  3. matlab 2015 积分,浅谈MATLAB在数值积分中的应用

    <浅谈MATLAB在数值积分中的应用.doc>由会员分享,可免费在线阅读全文,更多与<浅谈MATLAB在数值积分中的应用>相关文档资源请在帮帮文库(www.woc88.com) ...

  4. 浅谈Spark应用程序的性能调优

    浅谈Spark应用程序的性能调优 :http://geek.csdn.net/news/detail/51819 下面列出的这些API会导致Shuffle操作,是数据倾斜可能发生的关键点所在  1. ...

  5. 浅谈surging服务引擎中的rabbitmq组件和容器化部署

    1.前言 上个星期完成了surging 的0.9.0.1 更新工作,此版本通过nuget下载引擎组件,下载后,无需通过代码build集成,引擎会通过Sidecar模式自动扫描装配异构组件来构建服务引擎 ...

  6. python算法程序_浅谈python常用程序算法

    一.冒泡排序: 1.冒泡排序是将无序的数字排列成从小到大的有序组合: 过程:对相邻的两个元素进行比较,对不符合要求的数据进行交换,最后达到数据有序的过程. 规律: 1.冒泡排序的趟数时固定的:n-1 ...

  7. 浅谈数字媒体艺术中的技术应用-3-工具介绍(二)

    上一篇文章介绍了数据可视化工具Processing和OpenFrameWorks,以及开源电子硬件Arduino和RaspberryPI.这一篇继续把剩下的几个工具进行逐一的介绍. 商用游戏引擎:Un ...

  8. 浅谈小学语文教学中的读

    浅谈小学语文教学中的读 毛登凤<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" ...

  9. 转 浅谈游戏辅助程序的制作

    标 题: 浅谈游戏辅助程序的制作[原创] 作 者: qINGfENG 时 间: 2006-04-17,21:06 链 接: http://bbs.pediy.com/showthread.php?t= ...

  10. 关于计算机素养论文,浅谈计算机专业教学中如何提高学生计算机素养

    [摘 要] 计算机素养在培养学生核心素养方面具有重要意义,因此在计算机教学中,教师应结合计算机专业教学习现状,结合学生的全面发展需求和未来的职业规划,转变自身传统的教学观念,运用现代化的教学思维,将新 ...

最新文章

  1. 百度API地图 ,房产频道的标注方法
  2. C# 实现单线程线程池并调用实例
  3. 一个普通ERROR 1135 (HY000)错误引发的血案:
  4. [Linux]history 显示命令执行的时间
  5. oracle plan_table,Oracle 执行计划 提示 'PLAN_TABLE' is old version 解决方法
  6. java ssh客户端_简单的Java SSH客户端
  7. linux分割图片软件,桌面应用|5 种拆分 Linux 终端的方法
  8. CentOS系统yum源使用报错:Error: Cannot retrieve repository metadata
  9. 【解决】client does not support authentication
  10. 数据库还原的多种方式
  11. python操作 SVN中文乱码问题
  12. mui+html5+实现扫描二维码操作
  13. Unity入门操作_2D动画播放_038
  14. spyder快捷键大全
  15. VBA清除除第一行之外所有表格的数据
  16. 不同计算机用户的区别是什么意思,电脑操作系统的32位和64位分别是什么意思?有什么区别?...
  17. IDEA 要自己主动装的插件(等我安新的了再更新)
  18. 【狼人杀】初阶教学——基本规则
  19. GPS参数提取与轨迹重现实验
  20. 电脑版微信dat文件用什么软件打开

热门文章

  1. scala zip--拉链操作入门
  2. Java内存模型探秘
  3. Mysql--Auto_increment详解
  4. Spring Boot与Docker(一):微服务架构和容器化概述
  5. SQLite升级数据库:
  6. 用Java开源项目JOONE实现人工智能编程
  7. UDT协议实现分析——数据发送控制
  8. Android 打包keysotre文件
  9. 周末献礼 MyVoix2.0.js 麦克风波形绘制(一)
  10. 软件测试——软件测试的实质