对象的释放Dispose和Close对比

C#内存释放的几个方法对比:html

image

而Close与Dispose这两种方法的区别在于,调用完了对象的Close方法后,此对象有可能被从新进行使用;而Dispose方法来讲,此对象所占有的资源须要被标记为无用了,也就是此对象被销毁了,不能再被使用。程序员

在实现Dispose方法的时候,必定要加上“GC.SuppressFinalize( this )”语句,避免再让GC调用对象的析构函数。数据库

 

=====================================================================函数

 

C#实现IDispose接口

 

.net的GC机制有两个问题:首先GC并不能释放全部资源,它更不能释放非托管资源。其次,GC也不是实时的,全部GC存在不肯定性。
为了解决这个问题donet提供了析构函数学习

复制代码
复制代码
public class TestClass : System.IDisposable
{
    //供程序员显式调用的Dispose方法
    public void Dispose()
    {
        //调用带参数的Dispose方法,释放托管和非托管资源
        Dispose(true);
        //手动调用了Dispose释放资源,那么析构函数就是没必要要的了,这里阻止GC调用析构函数
        System.GC.SuppressFinalize(this);
    }

    //protected的Dispose方法,保证不会被外部调用。
    //传入bool值disposing以肯定是否释放托管资源
    protected void Dispose(bool disposing)
    {
        if (disposing)
        {
            ///TODO:在这里加入清理"托管资源"的代码,应该是xxx.Dispose();
        }
        ///TODO:在这里加入清理"非托管资源"的代码
    }

    //供GC调用的析构函数
    ~TestClass()
    {
        Dispose(false);//释放非托管资源
    }
}
复制代码
复制代码

而即便咱们忘记了在合适的时候调用Dispose,GC也会在释放对象的时候帮咱们清理非托管资源的。GC所充当的角色只是一种保障手段,它应该充当这种角色,咱们不能过度依赖它。实际上,在较大的模块退出时咱们还应该及时地手动调用GC.Collect进行垃圾回收。网站

为何实现IDisposable接口的类的对象,由于.net CLR是采用GC(垃圾回收器)机制管理内存,不想C++语言那样,能保证对象的析构函数在做用域结束时被老是被自动调用,有时若是程序运行的过程当中一直没有知足启动GC的条件,则可能GC一次也没启动。 这样,若是一个类须要占用重要资源,就应该实现IDisposable接口,或者使用另外一种简捷的方式:使用Using,如:this

Using(MyClass myObj = new MyClass())url

{ ... }spa

对于没有实现IDisposable接口的,也就没什么Dispose方法,但他们的Finalize一样不能保证被调用。

Using(MyClass myObj = new MyClass())

{ ... }

是一种好方法,可是只有MyClass实现了IDisposable接口才能这样写.

 

IDispose模式在C++中用的不少,用来清理资源,而在C#里,资源分为托管和非托管两种,托管资源是由C#的CLR帮助咱们清理的,它是经过调用对象的析构函数完成的对象释放工做,而对于非托管系统来讲,则须要咱们本身来释放,例如数据库链接对象,这就须要咱们手动去调用它的Dispose()方法来实现对象它的释放,事实上,Dispose()内容到底作了什么事,咱们并不清楚,固然这就是面向对象,它不但愿你关系实现的细节,呵!

对于咱们开发人员来讲,在了解它怎么用以后,总会对它如何实现的产生兴趣,下面,我将把C#里实现IDispose模式的代码展示出来,你们一块儿来学习一下,事实上,它的使用场合也不少的,当咱们手动对网站,数据库做封装时,都会用的到,下面看一下代码:

复制代码
复制代码
 /// <summary>
    /// 实现IDisposable,对非托管系统进行资源回收
    /// </summary>
    public class IDisplosePattern : IDisposable
    {
        public void Dispose()
        {
            this.Dispose(true);////释放托管资源
            GC.SuppressFinalize(this);//请求系统不要调用指定对象的终结器. //该方法在对象头中设置一个位,系统在调用终结器时将检查这个位
        }

        protected virtual void Dispose(bool disposing)
        {
            if (!_isDisposed)//_isDisposed为false表示没有进行手动dispose
            {
                if (disposing)
                {
                    //清理托管资源
                }
                //清理非托管资源
            }
            _isDisposed = true;
        }

        private bool _isDisposed;

        ~IDisplosePattern()
        {
            this.Dispose(false);//释放非托管资源,托管资源由终极器本身完成了
        }
    }
复制代码
复制代码

经过上面的代码,咱们知道了,对于托管系统(C#的CLR为咱们管理的),直接经过~IDisplosePattern()方法进行释放,而~IDisplosePattern()这个方法什么时候被调用,咱们是不知道的,由于它是由CLR帮助咱们调用的,而咱们手动进行dispose方法时,它会调用dispose(true)这个重载方法,它会帮助咱们清理托管和非托管资源,如图:

 

出处:http://www.javashuo.com/article/p-erqhcpbw-gk.html

相关文章
相关标签/搜索