C#中析構(gòu)函數(shù)、Dispose、Close方法的區(qū)別
一、Close與Dispose這兩種方法的區(qū)別
調(diào)用完了對象的Close方法后,此對象有可能被重新進行使用;而Dispose方法來說,此對象所占有的資源需要被標記為無用了,也就是此對象要被銷毀,不能再被使用。例如常見.Net類庫中的SqlConnection這個類,當(dāng)調(diào)用完Close方法后,可以通過Open重新打開一個數(shù)據(jù)庫連接,當(dāng)徹底不用這個對象了就可以調(diào)用Dispose方法來標記此對象無用,等待GC回收。
二、三者的區(qū)別如圖
析構(gòu)函數(shù) | Dispose方法 | Close方法 | |
意義 | 銷毀對象 | 銷毀對象 | 關(guān)閉對象資源 |
調(diào)用方式 | 不能被顯示調(diào)用,在GC回收是被調(diào)用 | 需要顯示調(diào)用或者通過using語句 | 需要顯示調(diào)用 |
調(diào)用時機 | 不確定 | 確定,在顯示調(diào)用或者離開using程序塊 | 確定,在顯示調(diào)用時 |
三、析構(gòu)函數(shù) 和 Dispose 的說明
Dispose需要實現(xiàn)IDisposable接口。
Dispose由開發(fā)人員代碼調(diào)用,而析構(gòu)函數(shù)由GC自動調(diào)用。
Dispose方法應(yīng)釋放所有托管和非托管資源。而析構(gòu)函數(shù)只應(yīng)釋放非托管資源。因為析構(gòu)函數(shù)由GC來判斷調(diào)用,當(dāng)GC判斷某個對象不再需要的時候,則調(diào)用其析構(gòu)方法,這時候該對象中可能還包含有其他有用的托管資源。
Dispose方法結(jié)尾處加上代碼“GC.SuppressFinalize(this);”,即告訴GC不需要再調(diào)用該對象的析構(gòu)方法,否則,GC仍會在判斷該對象不再有用后調(diào)用其析構(gòu)方法,雖然程序不會出錯,但影響系統(tǒng)性能。
析構(gòu)函數(shù) 和 Dispose 釋放的資源應(yīng)該相同,這樣即使類使用者在沒有調(diào)用 Dispose 的情況下,資源也會在 Finalize 中得到釋放。
Finalize 不應(yīng)為 public。
通過系統(tǒng)GC頻繁的調(diào)用析構(gòu)方法來釋放資源會降低系統(tǒng)性能,所以推薦顯示調(diào)用Dispose方法。
有 Dispose 方法存在時,應(yīng)該調(diào)用它,因為 Finalize 釋放資源通常是很慢的。
四、Close函數(shù)的說明
Close 這個方法在不同的類中有不同的含義,并沒有任何規(guī)定要求 Close 具有特殊的含義,也就是說 Close 并不一定要釋放資源,您也可以讓 Close 方法表示“關(guān)門”。 不過,由于 Close 有“關(guān)”的意思,通常也把 Close 拿來釋放資源,這也是允許的。比如文件操作中,用 Close 釋放對象似乎比 Dispose 含義更準確,于是在設(shè)計類時,可以將 Close 設(shè)為 public,將 Dispose 設(shè)為 protected,然后由 Close 調(diào)用 Dispose。 也就是說 Close 表示什么意思,它會不會釋放資源,完全由類設(shè)計者決定。網(wǎng)上說“Close 調(diào)用 Dispose”這種方法是很片面的。在 SqlConnection 中 Close 只是表示關(guān)閉數(shù)據(jù)庫連接,并沒有釋放 SqlConnection 這個對象資源。 根據(jù)經(jīng)驗,Close 和 Dispose 同時存在的情況下(均為 public),Close 并不表示釋放資源,因為通常情況下,類設(shè)計者不應(yīng)該使用兩個 public 方法來釋放相同的資源。
五、析構(gòu)函數(shù)和Dispose方法實例
{
//前面我們說了析構(gòu)函數(shù)實際上是重寫了 System.Object 中的虛方法 Finalize, 默認情況下,一個類是沒有析構(gòu)函數(shù)的,也就是說,對象被垃圾回收時不會被調(diào)用Finalize方法
~BaseResource()
{
// 為了保持代碼的可讀性性和可維護性,千萬不要在這里寫釋放非托管資源的代碼
// 必須以Dispose(false)方式調(diào)用,以false告訴Dispose(bool disposing)函數(shù)是從垃圾回收器在調(diào)用Finalize時調(diào)用的
Dispose(false);
}
// 無法被客戶直接調(diào)用
// 如果 disposing 是 true, 那么這個方法是被客戶直接調(diào)用的,那么托管的,和非托管的資源都可以釋放
// 如果 disposing 是 false, 那么函數(shù)是從垃圾回收器在調(diào)用Finalize時調(diào)用的,此時不應(yīng)當(dāng)引用其他托管對象所以,只能釋放非托管資源
protected virtual void Dispose(bool disposing)
{
// 那么這個方法是被客戶直接調(diào)用的,那么托管的,和非托管的資源都可以釋放
if (disposing)
{
// 釋放 托管資源
OtherManagedObject.Dispose();
}
//釋放非托管資源
DoUnManagedObjectDispose();
// 那么這個方法是被客戶直接調(diào)用的,告訴垃圾回收器從Finalization隊列中清除自己,從而阻止垃圾回收器調(diào)用Finalize方法.
if (disposing)
GC.SuppressFinalize(this);
}
//可以被客戶直接調(diào)用
public void Dispose()
{
//必須以Dispose(true)方式調(diào)用,以true告訴Dispose(bool disposing)函數(shù)是被客戶直接調(diào)用的
Dispose(true);
}
}
上面的范例達到的目的:
1/ 如果客戶沒有調(diào)用Dispose(),未能及時釋放托管和非托管資源,那么在垃圾回收時,還有機會執(zhí)行Finalize(),釋放非托管資源,但是造成了非托管資源的未及時釋放的空閑浪費
2/ 如果客戶調(diào)用了Dispose(),就能及時釋放了托管和非托管資源,那么該對象被垃圾回收時,不回執(zhí)行Finalize(),提高了非托管資源的使用效率并提升了系統(tǒng)性能
最后:
如果您的類中使用了非托管資源,則要考慮提供Close方法,和Open方法。并在您的Dispose方法中先調(diào)用 Close方法。
在使用已經(jīng) 有類時,如SqlConnection。如果暫時不用這個連接,可以考慮用Close()方法。如果不用了就考慮調(diào)用Dispose()方法。
相關(guān)文章
c#編寫的高并發(fā)數(shù)據(jù)庫控制訪問代碼
往往大數(shù)據(jù)量,高并發(fā)時, 瓶頸都在數(shù)據(jù)庫上, 好多人都說用數(shù)據(jù)庫的復(fù)制,發(fā)布, 讀寫分離等技術(shù), 但主從數(shù)據(jù)庫之間同步時間有延遲.2015-03-03Unity實現(xiàn)鼠標點2D轉(zhuǎn)3D進行旋轉(zhuǎn)
這篇文章主要為大家詳細介紹了Unity實現(xiàn)鼠標點2D轉(zhuǎn)3D進行旋轉(zhuǎn),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2020-04-04SQL+C#實現(xiàn)獲得當(dāng)前月的第一天與最后一天
本文分享了SQL+C#獲得當(dāng)前月的第一天與最后一天的代碼實例,代碼簡潔,適合初學(xué)者參考。需要的朋友可以看下2016-12-12C#使用zxing/zbar/thoughtworkQRcode解析二維碼的示例代碼
zxing是谷歌開源的二維碼庫,zbar,thoughtworkQRcode也是開源的,三者之間比較各有優(yōu)劣,本文將通過一個案例demo源碼,帶來認識學(xué)習(xí)下這三者的實際解碼效果,感興趣的可以了解一下2023-07-07