using语句是C#中确保非托管资源及时释放的推荐方式,本质为try...finally语法糖,编译后保证Dispose()必调;实现IDisposable需遵循标准Dispose模式,区分托管与非托管清理,并防止重复释放。
在C#中,using语句是确保资源及时、安全释放的最常用且推荐的方式,其背后依赖的是IDisposable接口。它不是“自动垃圾回收”的替代品,而是专门用于显式释放**非托管资源**(如文件句柄、数据库连接、网络流、GDI对象等)或需要立即清理的托管资源。
using语句在编译后会被转换为try...finally结构,保证无论是否发生异常,Dispose()方法都会被执行。它不负责对象生命周期管理,也不影响GC行为,只负责调用一次Dispose()。
例如:
using (var file = new FileStream("log.txt", FileMode.Create))
{
file.Write(data, 0, data.Length);
} // ← 这里隐式调用 file.Dispose()等价于:
FileStream file = null;
try
{
file = new FileStream("log.txt", FileMode.Create);
file.Write(data, 0, data.Length);
}
finally
{
file?.Dispose(); // 编译器自动生成
}如果你的类持有非托管资源(如IntPtr),或封装了

IDisposable对象,应实现完整的Dispose模式——区分“托管清理”和“非托管清理”,并提供Dispose(bool)虚方法。多数情况下,你只需做托管清理,可省略finalizer。
典型实现要点:
private bool _disposed = false;,防止重复释放void Dispose(),调用Dispose(true)并抑制终结器(GC.SuppressFinalize(this))protected virtual void Dispose(bool disposing)
disposing为true时,释放托管资源(如调用_stream?.Dispose())disposing为false时(即从finalizer调用),只释放非托管资源(如Marshal.FreeHGlobal(_ptr))if (_disposed) throw new ObjectDisposedException(...);做状态校验很多人误以为using能“延长对象生命”或“避免NullReferenceException”,其实不然。它只是语法便利,资源释放时机由作用域决定。
StreamWriter写入已关闭流会抛ObjectDisposedException)IDisposable;struct虽可实现但无意义(栈上分配,离开作用域即销毁)using (var a = ..., var b = ...) { ... },它们按声明逆序被Dispose(b先于a)using var file = new FileStream(...);,作用域为当前代码块(如方法或if分支)await using配合IAsyncDisposable,适用于DbContext、HttpClient等纯托管、无外部句柄、不包装其他IDisposable对象的类,通常无需实现。比如一个只含字符串和int的DTO类,GC回收即可。
判断依据只有一个:该类是否直接或间接持有必须主动释放的资源? 如果答案是否定的,就别加IDisposable——过度实现反而增加维护成本和误用风险。
基本上就这些。核心就两点:用using确保Dispose必达;实现IDisposable时守住“谁创建谁释放”和“只释放一次”的原则。不复杂,但容易忽略细节。