IDisposable接口用于释放非托管资源,核心是实现Dispose方法并区分托管与非托管资源。1. 只含void Dispose()方法,配合using语句自动调用;2. 持有非托管资源时需实现完整Dispose模式,包括Dispose(bool)、析构函数和GC.SuppressFinalize;3. Dispose(bool disposing)中,true时释放托管资源,false时仅清理非托管资源;4. 析构函数作为安全网,仅在直接管理非托管资源时需要;5. 可被继承的类应提供protected virtual Dispose(bool)供子类重写;6. 使用using块确保及时释放资源,防止泄漏。正确实现可避免资源未释放问题。

IDisposable 接口是 C# 中用于释放非托管资源(如文件句柄、数据库连接、网络套接字等)的标准机制。它只包含一个方法:Dispose(),通过手动调用或使用 using 语句来确保资源被及时清理。
如果类持有非托管资源或实现了 IDisposable 的对象,就应实现 IDisposable 接口,避免资源泄漏。
基本用法:IDisposable 接口
定义:IDisposable 只有一个方法:
void Dispose();
常见用法是在 using 块中使用:
using (var file = File.Open("data.txt", FileMode.Open))
{
// 使用文件流
} // 自动调用 Dispose()
正确实现 Dispose 模式
当类直接管理非托管资源时,需要完整实现 Dispose 模式,包括析构函数和资源释放逻辑。
关键点:
- 实现
IDisposable接口 - 提供受保护的虚方法
Dispose(bool) - 避免重复释放
- 必要时添加析构函数(仅非托管资源)
标准实现模板:
public class MyClass : IDisposable
{
private IntPtr _handle; // 非托管资源示例
private FileStream _fileStream; // 托管资源
private bool _disposed = false;
public MyClass()
{
_handle = AllocateSomeNativeResource();
_fileStream = File.Open("log.txt", FileMode.Create);
}
protected virtual void Dispose(bool disposing)
{
if (_disposed) return;
if (disposing)
{
// 释放托管资源
_fileStream?.Dispose();
}
// 释放非托管资源
ReleaseNativeResource(_handle);
_handle = IntPtr.Zero;
_disposed = true;
}
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this); // 避免析构函数再次释放
}
~MyClass()
{
Dispose(false); // 不释放托管资源,仅清理非托管部分
}}
何时需要析构函数?
只有在直接持有非托管资源(如指针、句柄)时才需要析构函数。它作为“安全网”,防止用户忘记调用 Dispose。
注意:- 析构函数运行时机不确定,不应依赖它及时释放资源
- 一旦调用了
Dispose(),应调用GC.SuppressFinalize(this)避免重复处理
继承场景下的处理
若类可能被继承,Dispose(bool) 应声明为 protected virtual,子类可重写以添加自己的清理逻辑。
protected override void Dispose(bool disposing)
{
if (!_disposed)
{
if (disposing)
{
// 清理子类的托管资源
}
// 清理子类的非托管资源
base.Dispose(disposing);
}}
基本上就这些。核心是区分托管与非托管资源,合理组织释放逻辑,利用 using 确保调用,避免资源泄漏。










