浅谈C#托管程序中的资源释放问题

摘要:
最后,我开始写这篇文章。一位网友几次催促我,但我要么忙,要么被其他想法分散了注意力,以至于这篇文章很久没写了。今天,我终于可以平静下来完成它。是的,NET工具编写程序,不难发现它的优点是所使用的内存不需要释放,特别是在使用C#或VBNet时。因为程序占用的内存由系统管理,程序员不需要担心内存的释放。许多人已经从C语言或C++和其他语言转变过来

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

 

用了.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何时去检测和收集是不确定的,因此对象的析构函数调用时机也是不确定的。这里也暗藏了一个道理,就是在析构函数中去做一些资源的关闭和标记就不是很合理了,因为所占有的资源无法迅速地进行关闭或者标记为无用。

 

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

 

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

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

 

 

浅谈C#托管程序中的资源释放问题第1张浅谈C#托管程序中的资源释放问题第2张/**//// 
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张    
///
 The class to show three disposal function
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第6张    
/// 

浅谈C#托管程序中的资源释放问题第7张
浅谈C#托管程序中的资源释放问题第7张    
public class DisposeClass:IDisposable
浅谈C#托管程序中的资源释放问题第7张
浅谈C#托管程序中的资源释放问题第1张浅谈C#托管程序中的资源释放问题第2张    
浅谈C#托管程序中的资源释放问题第12张
{
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张        
public void
 Close()
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第16张浅谈C#托管程序中的资源释放问题第17张        
浅谈C#托管程序中的资源释放问题第12张
{
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张            Debug.WriteLine( 
"Close called!"
 );
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第22张        }

浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张 
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张        
~DisposeClass()
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第16张浅谈C#托管程序中的资源释放问题第17张        
浅谈C#托管程序中的资源释放问题第12张
{
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张            Debug.WriteLine( 
"Destructor called!"
 );
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第22张        }

浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张 
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第38张浅谈C#托管程序中的资源释放问题第39张        
IDisposable Members#region IDisposable Members
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张 
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张        
public void Dispose()
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第16张浅谈C#托管程序中的资源释放问题第17张        
浅谈C#托管程序中的资源释放问题第12张
{
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张            
// TODO:  Add DisposeClass.Dispose implementation

浅谈C#托管程序中的资源释放问题第3张

浅谈C#托管程序中的资源释放问题第3张            Debug.WriteLine( 
"Dispose called!" );
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第22张        }

浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张 
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第22张        
#endregion

浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第6张    }

浅谈C#托管程序中的资源释放问题第7张
浅谈C#托管程序中的资源释放问题第7张

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

        privatevoid Create()

        {

            DisposeClass myClass = new DisposeClass();

        }

 

        privatevoid 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如下:

   

浅谈C#托管程序中的资源释放问题第1张浅谈C#托管程序中的资源释放问题第2张 /**//// 
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张    
///
 The class to show three disposal function
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第6张    
/// 

浅谈C#托管程序中的资源释放问题第7张
浅谈C#托管程序中的资源释放问题第7张    
public class DisposeClass:IDisposable
浅谈C#托管程序中的资源释放问题第7张
浅谈C#托管程序中的资源释放问题第1张浅谈C#托管程序中的资源释放问题第2张    
浅谈C#托管程序中的资源释放问题第12张
{
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张        
public void
 Close()
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第16张浅谈C#托管程序中的资源释放问题第17张        
浅谈C#托管程序中的资源释放问题第12张
{
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张            Debug.WriteLine( 
"Close called!"
 );
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第22张        }

浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张 
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张        
~DisposeClass()
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第16张浅谈C#托管程序中的资源释放问题第17张        
浅谈C#托管程序中的资源释放问题第12张
{
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张            Debug.WriteLine( 
"Destructor called!"
 );
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第22张        }

浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张 
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第38张浅谈C#托管程序中的资源释放问题第39张        
IDisposable Members#region IDisposable Members
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张 
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张        
public void Dispose()
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第16张浅谈C#托管程序中的资源释放问题第17张        
浅谈C#托管程序中的资源释放问题第12张
{
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张            
// TODO:  Add DisposeClass.Dispose implementation

浅谈C#托管程序中的资源释放问题第3张

浅谈C#托管程序中的资源释放问题第3张            Debug.WriteLine( 
"Dispose called!" );
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张            GC.SuppressFinalize( 
this
 );
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第22张        }

浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第3张 
浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第22张        
#endregion

浅谈C#托管程序中的资源释放问题第3张
浅谈C#托管程序中的资源释放问题第6张    }

浅谈C#托管程序中的资源释放问题第7张
浅谈C#托管程序中的资源释放问题第7张

 

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

        privatevoid Run()

        {

            using( DisposeClass myClass = new DisposeClass() )

            {

                //other operation here

            }

        }

 

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

免责声明:文章转载自《浅谈C#托管程序中的资源释放问题》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇Gradle配置最佳实践小程序canvas生成二维码图片踩的坑下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

C#中Bitmap类 对图像の操作 可检测图片完整性

try { Bitmap bm = new Bitmap(pics[ip]); BitmapToBytes(bm).Reverse().Take(2); } catch (Exception ex) { string imgName = Path.GetFileName(pics[ip]); File.Move(pics[ip], txtImgDir...

C#中的String类2

                          深入C# String类 C#中的String类 他是专门处理字符串的(String),他在System的命名空间下,在C#中我们使用的是string 小写的string只是大写的String的一个别名(外号)使用大写和小写都是一样的 Using  == import 常用的字符串处理法 Java中常用的...

为什么使用C#开发软件的公司和程序员都很少?

特别是在国内, 在职位搜索网站indeed,职位关键字与数量的对应关系如下(搜索时实时数据): Java - 49624 Python - 16802 JavaScript - 25452 C ++ - 36569 C# - 14267 很显然,C#相关的职位数量是明显落后的,这是为什么呢? C#由编程界的神级人物Anders Hejlsberg主持...

C#: 通过html调用WinForm 。。。。。

完整测试代码:Form1.cs 代码 using System;using System.Collections.Generic;using System.ComponentModel;using System.Data;using System.Drawing;using System.Text;using System.Windows.Forms;n...

基于ASP.NET MVC(C#)和Quartz.Net组件实现的定时执行任务调度

http://www.cnblogs.com/bobositlife/p/aspnet-mvc-csharp-quartz-net-timer-task-scheduler.html 在之前的文章《推荐一个简单、轻量、功能非常强大的C#/ASP.NET定时任务执行管理器组件–FluentScheduler》和《简单、轻量、功能非常强大的C#/ASP.NET...

C#语音朗读文本 — TTS的实现

TTS, Text To Speech的缩写,是使用语音朗读文本的技术。目前,在国内应用较多的是排队叫号系统 Windows 平台的TTS,通常使用的是微软自带的 Speech API。 Windows XP 自带的是 Speech API 5.1,默认是只支持英文的。如果要朗读中文,需要安装LangPack,里面支持简体中文,不过,是个男生的发音。...