.NET 清理非托管资源
- Dispose
-
- 设计原则
应用程序或类库应只允许一个线程拥有资源的生存期,并且应在不再需要资源时调用 Dispose。根据资源的不同,在处置资源时进行异步线程访问可能会带来安全风险。开发人员应仔细检查自己的代码,以确定最佳的方法来强制线程安全。
-
- 代码示例
public class BaseResource: IDisposable
{
// 非托管资源
private IntPtr handle;
// 托管资源
private Component Components;
// 对象是否已被释放的标志
private bool disposed = false;
public BaseResource() { }
// 释放资源,对外开放的可调用的方法
public void Dispose()
{
// 释放资源
Dispose(true);
// 指示在析构函数中跳过垃圾回收
GC.SuppressFinalize(this);
}
// 释放资源,如果disposing为true,释放所有的托管资源和非托管资源,如果为false,则仅仅释放非托管资源,这主要是为了避免在析构函数中重复2次进行垃圾回收
protected virtual void Dispose(bool disposing)
{
// 检查该对象是否已经被释放了
if(!this.disposed)
{
if(disposing)
{
// 释放托管资源
Components.Dispose();
}
// 释放非托管资源
CloseHandle(handle);
handle = IntPtr.Zero;
}
// 标记该对象为已被释放的对象
disposed = true;
}
// 析构函数,又名终结器
~BaseResource()
{
// 释放非托管资源,在调用终结器方法时系统自动会对托管的资源进行垃圾回收
Dispose(false);
}
// 允许多次调用Dispose,但会抛出异常publicvoid DoSomething()
{
if(this.disposed)
{
thrownew ObjectDisposedException();
}
}
}
|
-
- 实现 Close 方法
对于类型来说,若调用 Close 方法比调用 Dispose 方法更容易,则可以向基类型添加一个公共 Close 方法。Close 方法又会调用没有参数的 Dispose 方法,该方法可以执行正确的清理操作。在基础类库中的所有类的Close方法都是基于该原理构造的。
public void Close()
{
// 释放资源
Dispose();
}
|
- Finalize(终结器)
对于您的应用程序创建的大多数对象,可以依靠 .NET Framework 的垃圾回收器隐式地执行所有必要的内存管理任务。但是,在您创建封装非托管资源的对象时,当您在应用程序中使用完这些非托管资源之后,您必须显式地释放它们。
虽然垃圾回收器可以跟踪封装非托管资源的对象的生存期,但它不了解具体如何清理这些资源。对于这些类型的对象,.NET Framework 提供 Object.Finalize 方法,它允许对象在垃圾回收器回收该对象使用的内存时适当清理其非托管资源。但是对托管对象就不应该实现 Finalize方法,因为垃圾回收器会自动清理托管资源。
默认情况下,Finalize 方法不执行任何操作。如果您要让垃圾回收器在回收对象的内存之前对对象执行清理操作,您必须在类中重写 Finalize 方法。但是在 C# 或 C++ 编程语言中无法重写 Finalize 方法,所以在 C# 中可使用析构函数语法实现 Finalize 方法。
Finalize 方法主要是在未能调用 Dispose 方法的情况下充当防护措施来清理资源。
实现 Finalize 方法或析构函数对性能可能会有负面影响,因此应避免不必要地使用它们。
用 Finalize 方法回收对象使用的内存需要至少两次垃圾回收。当垃圾回收器执行回收时,它只回收没有终结器的不可访问对象的内存。这时,它不能回收具有终结器的不可访问对象。它改为将这些对象的项从终止队列中移除并将它们放置在标为准备终止的对象列表中。该列表中的项指向托管堆中准备被调用其终止代码的对象。垃圾回收器为此列表中的对象调用 Finalize 方法,然后,将这些项从列表中移除。后来的垃圾回收将确定终止的对象确实是垃圾,因为标为准备终止对象的列表中的项不再指向它们。在后来的垃圾回收中,实际上回收了对象的内存。
- 封装资源对象
如果您要编写代码,而该代码使用一个封装资源的对象,您应该确保在使用完该对象时调用该对象的 Dispose 方法。
-
- 封装方式
-
- using语句
- try/finally块