说说非托管资源的回收

释放未托管的资源有两种方法

1、析构函数

2、实现System.IDisposable接口

一、析构函数

构造函数可以指定必须在创建类的实例时进行的某些操作,在垃圾收集器删除对象时,也可以调用析构函数。析构函数初看起来似乎是放置释放未托管资源、执行一般清理操作的代码的最佳地方。但是,事情并不是如此简单。由于垃圾回收器的运行规则决定了,不能在析构函数中放置需要在某一时刻运行的代码,如果对象占用了宝贵而重要的资源,应尽可能快地释放这些资源,此时就不能等待垃圾收集器来释放了.

实例

C# 代码   复制

using System;

using System.Collections.Generic;

using System.Linq;

using System.Text;

namespace MemRelease

{
    class Program
    {
        ~Program()
        {
            // Orders.
        }

        static void Main(string[] args)
        {
        }
    }
}

在IL DASM中,你会发现并没有这个析构的方法。C#编译器在编译析构函数时,会隐式地把析构函数的代码编译为Finalize()方法的对应代码,确保执行父类的Finalize()方法 看下这段代码中对于析构函数的编译:

C# 代码   复制

.method family hidebysig virtual instance void

        Finalize() cil managed

{
  // Code size       14 (0xe)
  .maxstack  1
  .try
  {
    IL_0000:  nop
    IL_0001:  nop
    IL_0002:  leave.s    IL_000c
  }  // end .try
  finally
  {
    IL_0004:  ldarg.0
    IL_0005:  call       instance void [mscorlib]System.Object::Finalize()
    IL_000a:  nop
    IL_000b:  endfinally
  }  // end handler
  IL_000c:  nop
  IL_000d:  ret
} // end of method Program::Finalize

使用析构函数来释放资源有几个问题

1、与C++析构函数相比,C#析构函数的问题是他们的不确定性。在删除C++对象时,其析构函数会立即执行,但是由于垃圾收集器的工作方式,无法确定C#对象的析构函数何时执行。

2、C#析构函数的执行会延迟对象最终从内存中删除的时间。有析构函数的对象需要2次处理才能删除:第一次调用析构函数时,没有删除对象,第二次调用才真正删除对象。

二、IDisposable接口

IDisposable接口定义了一个模式,为释放未托管的资源提供了确定的机制,并避免产生析构函数固有的与垃圾函数器相关的问题。IDisposable接口声明了一个方法Dispose(),它不带参数,返回void。

1、MSDN建议按照下面的模式实现IDisposable接口

C# 代码   复制

 public class Foo: IDisposable

 {
     public void Dispose()
     {
        Dispose(true);
        GC.SuppressFinalize(this);
     }

     protected virtual void Dispose(bool disposing)
     {
        if (!m_disposed)
        {
            if (disposing)
            {
               // Release managed resources
            }

            // Release unmanaged resources

            m_disposed = true;
        }
     }

     ~Foo()
     {
        Dispose(false);
     }

     private bool m_disposed;
 }

在.NET的对象中实际上有两个用于释放资源的函数:Dispose和Finalize

(1)、Finalize的目的是用于释放非托管的资源,而Dispose是用于释放所有资源,包括托管的和非托管的

(2)、void Dispose(bool disposing)函数通过一个disposing参数来区别当前是否是被Dispose()调用

如果是被Dispose()调用,那么需要同时释放托管和非托管的资源。如果是被~Foo()(也就是C#的Finalize())调用了,那么只需要释放非托管的资源即可。

(3)、Dispose()函数是被其它代码显式调用并要求释放资源的,而Finalize是被GC调用的

在GC调用的时候Foo所引用的其它托管对象可能还不需要被销毁,并且即使要销毁,也会由GC来调用。因此在Finalize中只需要释放非托管资源即可。另外一方面,由于在Dispose()中已经释放了托管和非托管的资源,因此在对象被GC回收时再次调用Finalize是没有必要的,所以在Dispose()中调用GC.SuppressFinalize(this)避免重复调用Finalize。

然而,即使重复调用Finalize和Dispose也是不存在问题的,因为有变量m_disposed的存在,资源只会被释放一次,多余的调用会被忽略过去。

 

Finalize、Dispose保证了

(1)、 Finalize只释放非托管资源;

(2)、 Dispose释放托管和非托管资源;

(3)、 重复调用Finalize和Dispose是没有问题的;

(4)、 Finalize和Dispose共享相同的资源释放策略,因此他们之间也是没有冲突的。

2、IDisposable例子

C# 代码   复制

namespace 资源回收

{
    class Program
    {
        static void Main(string[] args)
        {
            //使用using对实现IDisposable的类了进行资源管理
/*拿到一个对象的时候,首先判断这个对象是否实现了IDisposable接口,如果实现了,最好就用using包裹住这个对象,保证这个对象用完之后被释放掉,否则很可能出现资源泄露的问题
*/
            using (Telphone t1 = new Telphone())
            {
                t1.Open();
                t1.Speak("hello");
                t1.Bomb();
                //t1.Dispose();//如果在这里调用了Dispose()方法释放资源,那么在执行t1.Open()方法就出错,电话线已经被剪断了,无法再打电话了
                t1.Open();
                t1.Speak("I am back!");
            }//代码执行到这里后,就会调用Dispose方法来进行资源回收
            Console.ReadKey();
        }
    }
    /// <summary>
    /// Telphone类实现了IDisposable接口
    /// </summary>
    class Telphone : IDisposable
    {
        /// <summary>
        /// 电话状态
        /// </summary>
        private TelphoneState state;
        /// <summary>
        /// 打电话
        /// </summary>
        public void Open()
        {
            if (state == TelphoneState.Disposed)
            {
                throw new Exception("电话线已经被剪断,无法打开!");
            }
            state = TelphoneState.Open;
            Console.WriteLine("拿起电话");
        }
        /// <summary>
        /// 说话
        /// </summary>
        /// <param name="s">说话内容</param>
        public void Speak(string s)
        {
            if (state != TelphoneState.Open)
            {
                throw new Exception("没有连接");
            }
            Console.WriteLine(s);
        }
        /// <summary>
        /// 挂掉电话
        /// </summary>
        public void Bomb()
        {
            state = TelphoneState.Close;
            Console.WriteLine("挂掉电话");
        }
        #region IDisposable 成员
        /// <summary>
        /// 实现IDisposable接口中的Dispose()方法来释放非托管资源
        /// 如何释放非托管资源由程序自己定
        /// </summary>
        public void Dispose()
        {
            if (state == TelphoneState.Open)
            {
                Bomb();//挂掉电话
            }
            state = TelphoneState.Disposed;
            Console.WriteLine("剪断电话线");
        }
        #endregion
    }
    /// <summary>
    /// 电话状态枚举
    /// </summary>
    enum TelphoneState
    {
        Open, Close, Disposed
    }
}

程序运行结果:

 

三、析构函数和IDisposable混合调用的例子

C# 代码   复制

public class ResourceHolder : IDisposable

{
      private bool isDispose = false;

      // 显示调用的Dispose方法
  public void Dispose()
      {
           Dispose(true);
          GC.SuppressFinalize(this);
       }

       // 实际的清除方法
  protected virtual void Dispose(bool disposing)
      {
            if (!isDisposed)
           {
              if (disposing)
           {
                      // 这里执行清除托管对象的操作.
                  }
                  // 这里执行清除非托管对象的操作
            }

         isDisposed=true;
      }

      // 析构函数
      ~ResourceHolder()
      {
            Dispose (false);
      }
}
时间: 2024-11-07 04:21:18

说说非托管资源的回收的相关文章

.net非托管资源的回收

释放未托管的资源有两种方法 1.析构函数 2.实现System.IDisposable接口 一.析构函数 构造函数可以指定必须在创建类的实例时进行的某些操作,在垃圾收集器删除对象时,也可以调用析构函数.析构函数初看起来似乎是放置释放未托管资源.执行一般清理操作的代码的最佳地方.但是,事情并不是如此简单.由于垃圾回收器的运行规则决定了,不能在析构函数中放置需要在某一时刻运行的代码,如果对象占用了宝贵而重要的资源,应尽可能快地释放这些资源,此时就不能等待垃圾收集器来释放了. 实例 C# 代码   复

C#编程(七十四)----------释放非托管资源

释放非托管资源 在介绍释放非托管资源的时候,我觉得有必要先来认识一下啥叫非托管资源,既然有非托管资源,肯定有托管资源. 托管资源指的是.net可以自棕进行回收的资源,主要是指托管堆上分配的内存资源.托管资源的回收工作是不需要人工干预的,有.net运行库在合适的调用垃圾回收器进行回收. 非托管资源指的是.net不知道如何回收的资源,最常见的一类非托管资源是包装操作系统资源的对象,例如文件,窗口,网络连接,数据库连接,画刷,图标等.这类资源,垃圾回收器在清理的时候会调用Object.Finalize

[转]在C#中使用托管资源和非托管资源的区别,以及怎样手动释放非托管资源:

托管资源指的是.NET可以自动进行回收的资源,主要是指托管堆上分配的内存资源.托管资源的回收工作是不需要人工干预的,有.NET运行库在合适调用垃圾回收器进行回收. 非托管资源指的是.NET不知道如何回收的资源,最常见的一类非托管资源是包装操作系统资源的对象,例如文件,窗口,网络连接,数据库连接,画刷,图标等.这类资源,垃圾回收器在清理的时候会调用Object.Finalize()方法.默认情况下,方法是空的,对于非托管对象,需要在此方法中编写回收非托管资源的代码,以便垃圾回收器正确回收资源. 在

.NET对象的创建、垃圾回收、非托管资源的手动处理

本篇用来梳理对象的创建.垃圾的回收,以及非托管资源的手动处理. →首先运行应用程序,创建一个Windows进程. →CLR创建一块连续的虚拟地址空间,这个地址空间就是托管堆.而且,这个地址空间最初并没有对应的物理存储空间. 虚拟地址空间分成2段.一个区段是普通堆,也叫GC堆,大小小于85000字节的引用类型对象的实例被分配在这里:另一个是大对象堆,大小大于等于85000字节的引用类型对象的实例被分配在这里. 对于客户端应用程序,每个区段的大小大致是16MB:对于服务端应用程序,每个区段的大小大致

.NET的堆和栈04,对托管和非托管资源的垃圾回收以及内存分配

在" .NET的堆和栈01,基本概念.值类型内存分配"中,了解了"堆"和"栈"的基本概念,以及值类型的内存分配.我们知道:当执行一个方法的时候,值类型实例会在"栈"上分配内存,而引用类型实例会在"堆"上分配内存,当方法执行完毕,"栈"上的实例由操作系统自动释放,"堆"上的实例由.NET Framework的GC进行回收. 在" .NET的堆和栈02,值类型和

C# 托管资源和非托管资源

托管资源指的是.NET可以自动进行回收的资源,主要是指托管堆上分配的内存资源.托管资源的回收工作是不需要人工干预的,有.NET运行库在合适调用垃圾回收器进行回收. 非托管资源指的是.NET不知道如何回收的资源,最常见的一类非托管资源是包装操作系统资源的对象,例如文件,窗口,网络连接,数据库连接,画刷,图标等.这类资源,垃圾回收器在清理的时候会调用Object.Finalize()方法.默认情况下,方法是空的,对于非托管对象,需要在此方法中编写回收非托管资源的代码,以便垃圾回收器正确回收资源. C

托管资源和非托管资源

在.net 编程环境中,系统的资源分为托管资源和非托管资源. 对于托管的资源的回收工作,是不需要人工干预回收的,而且你也无法干预他们的回收,所能够做的只是了解.net CLR如何做这些操作.也就是说对于您的应用程序创建的大多数对象,可以依靠 .NET Framework 的垃圾回收器隐式地执行所有必要的内存管理任务.        资源分为两种,托管的内存资源,这是不需要我们操心的,系统已经为我们进行管理了:那么对于非托管的资源,这里再重申一下,就是Stream,数据库的连接,GDI+的相关对象

.net 资源释放(托管资源和非托管资源)

1.托管资源 像int.float.DateTime等都是托管资源:net中80%的资源都是托管资源: 托管资源的回收通过GC(垃圾回收器)自动释放分配给该对象的内存,但无法预测进行垃圾回收的时间,我们无法控制系统在什么时间回收资源. 2.非托管资源 像ApplicationContext,Brush,Component,ComponentDesigner,Container,Context,Cursor,FileStream,Font,Icon,Image,Matrix,Object,Odbc

编写高质量代码改善C#程序的157个建议——建议50:在Dispose模式中应区别对待托管资源和非托管资源

建议50:在Dispose模式中应区别对待托管资源和非托管资源 真正资源释放代码的那个虚方法是带一个bool参数的,带这个参数,是因为我们在资源释放时要区别对待托管资源和非托管资源. 提供给调用者调用的显式释放资源的无参Dispose方法中,调用参数是true: public void Dispose() { //必须为true Dispose(true); //省略其他代码 } 这表明,这时候代码要同时处理托管资源和非托管资源. 在供垃圾回收器调用的隐式清理资源的终结器中,调用的是false: