探討C#中Dispose方法與Close方法的區(qū)別詳解
群里有人問,怎樣直接清空堆里的string值。有人建議直接用Dispose()方法;Dispose()銷毀了對象,是一種垃圾回收機制。(這里用using或許更好)
當我們開發(fā)C#代碼的時候,經(jīng)常碰到一個問題,有些class提供Close(),有些class提供Dispose(),那么Dispose和Close到底有什么區(qū)別?
在這里,要明確一下C#程序(或者說.NET)中的資源。簡單的說來,C#中的每一個類型都代表一種資源,而資源又分為兩類:
托管資源:由CLR管理分配和釋放的資源,即由CLR里new出來的對象;
非托管資源:不受CLR管理的對象,windows內(nèi)核對象,如文件、數(shù)據(jù)庫連接、套接字、COM對象等;毫無例外地,如果我們的類型使用到了非托管資源,或者需要顯式釋放的托管資源,那么,就需要讓類型繼承接口IDisposable。這相當于是告訴調(diào)用者,該類型是需要顯式釋放資源的,你需要調(diào)用我的Dispose方法。
首先,Dispose和Close基本上應該是一樣的。Close是為了那些不熟悉Dispose的開發(fā)者設計的。因為基本上所有的developer都知道Close是干嗎的(特別是對于那些有C++背景的developer)。
但是當我們寫code時候,如果要實現(xiàn)Close和Dispose的時候,要注意Close和Dispose的設計模式。.net的一些class只提供Close,而且派生自IDisposable,并且隱藏了Dispose方法。是不是覺得很不明白了?
對這些class來說,關鍵在于它們顯式的(explicitly)實現(xiàn)了IDisposable。對于隱式實現(xiàn)來說,你只需要調(diào)用"newA().Dispose()",但是對于顯式實現(xiàn)來說,Dispose不會是這個class的成員函數(shù)。唯一的調(diào)用方式是你先要cast到IDisposable才行。(“new A().Dispose()”編譯不過,但是“((IDisposable)newA()).Dispose()”可以編譯過)。所以這樣就符合了設計的要求:提供Close(),隱藏Dispose(),并且實現(xiàn)了IDisposable接口。
在.net的framework里,Close()被設計成public的,并且在Close()里面call被隱藏的Dispose();Dispose()去call另一個virtual的Dispose(bool)函數(shù)。所以如果你從這個class繼承,你就必須實現(xiàn)Dispose(bool)方法。
用者call Close()的時候就會call到你重載的那個Dispose(bool)方法去釋放資源。
下面是一個標準的繼承了IDisposable接口類型的實現(xiàn)方式,這種實現(xiàn)我們稱之為Dispose模式:
public class SampleClass : IDisposable
{
//演示創(chuàng)建一個非托管資源
private IntPtr nativeResource = Marshal.AllocHGlobal(100);
//演示創(chuàng)建一個托管資源
private AnotherResource managedResource = new AnotherResource();
private bool disposed = false;
/// <summary>
/// 實現(xiàn)IDisposable中的Dispose方法
/// </summary>
public void Dispose()
{
//必須為true
Dispose(true);
//通知垃圾回收機制不再調(diào)用終結器(析構器)
GC.SuppressFinalize(this);
}
/// <summary>
/// 不是必要的,提供一個Close方法僅僅是為了更符合其他語言(如C++)的規(guī)范
/// </summary>
public void Close()
{
Dispose();
}
/// <summary>
/// 必須,以備程序員忘記了顯式調(diào)用Dispose方法
/// </summary>
~SampleClass()
{
//必須為false
Dispose(false);
}
/// <summary>
/// 非密封類修飾用protected virtual
/// 密封類修飾用private
/// </summary>
/// <param name="disposing"></param>
protected virtual void Dispose(bool disposing)
{
if (disposed)
{
return;
}
if (disposing)
{
// 清理托管資源
if (managedResource != null)
{
managedResource.Dispose();
managedResource = null;
}
}
// 清理非托管資源
if (nativeResource != IntPtr.Zero)
{
Marshal.FreeHGlobal(nativeResource);
nativeResource = IntPtr.Zero;
}
//讓類型知道自己已經(jīng)被釋放
disposed = true;
}
public void SamplePublicMethod()
{
if (disposed)
{
throw new ObjectDisposedException("SampleClass", "SampleClass is disposed");
}
//省略
}
}
相關文章
.Net WInform開發(fā)筆記(五)關于事件Event
我前面幾篇博客中提到過.net中的事件與Windows事件的區(qū)別,本文討論的是前者,也就是我們代碼中經(jīng)常用到的Event,感興趣的朋友可以了解下2013-01-01C# 關于爬取網(wǎng)站數(shù)據(jù)遇到csrf-token的分析與解決
這篇文章主要介紹了C# 關于爬取網(wǎng)站數(shù)據(jù)遇到csrf-token的分析與解決,幫助大家更好的理解和學習c#,感興趣的朋友可以了解下2021-01-01