詳解c# 接口IDisposable的用法
C#的每一個類型都代表一種資源,而資源又分為兩類:
- 托管資源 由CLR管理分配和釋放的資源,即從CLR里new出來的對象。
- 非托管資源 不受CLR管理的對象,如Windows內(nèi)核對象,或者文件、數(shù)據(jù)庫連接、套接字、COM對象等。
如果類型用到了非托管資源,或者需要顯式釋放托管資源,那么需要讓類型繼承接口IDisposable。記?。喝绻愋托枰@式釋放資源,那么一定要繼承IDisposable接口。如:
class SampleClass:IDisposable { private IntPtr nativeResource = Marshal.AllocHGlobal(100);//非托管資源 private Bitmap bitmap = new Bitmap(100, 100);//托管資源 private bool isDisposed = false; //實現(xiàn)IDisposable中的Dispose方法 public void Dispose( ) { Dispose(true); GC.SuppressFinalize(this);//通知垃圾回收器不用再調(diào)用終結(jié)器 } //不必要的方法,只是為了符合其他語言的規(guī)范 public void Close() { Dispose(); } //必須的,防止程序員忘記顯示調(diào)用Dispose方法(隱式清理) ~SampleClass() { Dispose(false); } //非密封類修飾用protected virtual,提醒子類必須實現(xiàn)自己的清理方法時注意到父類的清理工作 protected virtual void Dispose(bool isDisposing) { if(isDisposed) { return; } if(isDisposing) { //清理托管資源 if(bitmap != null) { bitmap.Dispose(); bitmap = null; } } //清理非托管資源 if(nativeResource!=IntPtr.Zero) { Marshal.FreeHGlobal(nativeResource); nativeResource = IntPtr.Zero; } isDisposed = false; } public void SamplePublicMethod() { if(isDisposed) { throw new ObjectDisposedException("SampleClass", "SampleClass is disposed"); } //代碼 } }
繼承IDisposable接口,可以使用using語法糖。在using語句代碼塊內(nèi),可以使用聲明的對象,當(dāng)語句離開代碼塊后,系統(tǒng)自動釋放資源:
//使用using方法,當(dāng)語句離開代碼塊后,using內(nèi)的對象自動釋放 using (SampleClass sample = new SampleClass()) { //…… } //以上代碼相當(dāng)于下面的代碼 SampleClass sample0 = new SampleClass(); try { //…… } finally { sample0.Dispose(); }
在SampleClass中,存在一個終結(jié)器(C++中叫析構(gòu)器)。其意義在于,調(diào)用者可能并不會主動調(diào)用Dispose方法,而終結(jié)器會被垃圾回收器調(diào)用調(diào)用,所以它作為釋放資源的補救措施。
在CLR中,每new一個對象時,就會為該對象在堆上分配內(nèi)存,如果不再被引用,就會回收它們的內(nèi)存。如果沒有實現(xiàn)IDisposable接口的類型對象,垃圾回收器會直接釋放對象所占內(nèi)存;如果實現(xiàn)了,每次創(chuàng)建對象時,CLR會將該對象的一個指針放到終結(jié)列表中,垃圾回收器在回收對象前會首先將終結(jié)列表中的指針放入一個freachable隊列。同時,CLR會分配一個線程管理freachable隊列,調(diào)用對象終結(jié)器,只有此時,對象才會被真正標(biāo)識為垃圾,并在下一次進行垃圾回收時釋放對象所占內(nèi)存。即:實現(xiàn)IDisposable接口的類型,至少要經(jīng)過兩次垃圾回收才能真正釋放掉內(nèi)存。其中Dispose方法中的GC.SuppressFinalize()方法用于在顯示釋放資源后,通知垃圾回收器不用再調(diào)用終結(jié)器(隱式回收)釋放資源。
在實現(xiàn)IDisposable接口時,其Dispose()方法并沒有做實際的清理工作,但提供了帶bool參數(shù)的受保護的虛方法。因為該類型可能被其他類繼承,如果子類實現(xiàn)自己的Dispose模式,受保護的虛方法可以提醒子類:在實現(xiàn)自己的清理方法時,需要注意父類的清理工作(base.Dispose方法)。
真正撰寫資源釋放代碼的虛方法有一個bool參數(shù),但是在顯示釋放資源(true)與隱式釋放資源(false)調(diào)用中傳入的參數(shù)不同。表明:隱式清理時,只需要處理非托管資源就行。托管資源中的普通類型不需要手動清理,而非普通類型需要手動清理。
Dispose模式設(shè)計思路:如果調(diào)用者顯示調(diào)用了Dispose方法,那么類型按部就班釋放自己的全部資源,然后通知垃圾回收器不需要再釋放(GC.SuppressFinalize()方法);而忘記調(diào)用Dispose方法,那么類型就假定自己的所有托管資源會全部交給垃圾回收器回收,不需要手動清理。
以上就是詳解c# 接口IDisposable的用法的詳細內(nèi)容,更多關(guān)于c# 接口IDisposable的用法的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C#采用OpenXml實現(xiàn)給word文檔添加文字
這篇文章主要介紹了C#采用OpenXml實現(xiàn)給word文檔添加文字的方法,包括了用法的實例分析,是非常實用的技巧,需要的朋友可以參考下2014-09-09C#實現(xiàn)導(dǎo)出List數(shù)據(jù)到xml文件的方法【附demo源碼下載】
這篇文章主要介紹了C#實現(xiàn)導(dǎo)出List數(shù)據(jù)到xml文件的方法,涉及C#針對list類及xml文件的相關(guān)操作技巧,并附帶完整demo源碼供讀者下載參考,需要的朋友可以參考下2016-08-08詳解如何通過wireshark實現(xiàn)捕獲C#上傳的圖片
這篇文章主要為大家詳細介紹了如何通過wireshark實現(xiàn)捕獲C#上傳的圖片,文中的示例代碼簡潔易懂,具有一定的學(xué)習(xí)價值,感興趣的小伙伴可以了解下2023-11-11利用C#實現(xiàn)將小數(shù)值四舍五入為整數(shù)
在項目的開發(fā)中,遇到一些除法計算內(nèi)容會產(chǎn)生小數(shù)值,但是又需要根據(jù)項目的實際情況將這些小數(shù)內(nèi)容化為整數(shù),所以本文為大家整理了C#實現(xiàn)將小數(shù)值四舍五入為整數(shù)的方法,希望對大家有所幫助2023-07-07