淺析C# Dispose方法模式的建立
這里我們向你講解的是在建立包含內(nèi)存以外資源的C# Dispose方法模式類型,特別是處置非內(nèi)存資源的時候,如何編寫自己的資源管理代碼呢?讓我們開始講解吧。
我們已經(jīng)知道了處置(由C# Dispose方法來做)那些占用非受控(unmanaged)資源的對象的重要性,現(xiàn)在應(yīng)該編寫資源管理代碼來處置那些包含非內(nèi)存資源的類型了。整個.NET框架組件都使用一個標(biāo)準(zhǔn)的模式來處理非內(nèi)存資源。使用你建立的類型的用戶也希望你遵循這個標(biāo)準(zhǔn)的模式。標(biāo)準(zhǔn)的處理模式的思想是這樣的:當(dāng)客戶端記得的時候使用IDisposable接口釋放你的非受控資源,當(dāng)客戶端忘記的時候防護(hù)性地使用終結(jié)器(finalizer)。它與垃圾收集器(Garbage Collector)一起工作,確保只在必要的時候該對象才受到與終結(jié)器相關(guān)的性能影響。這是處理非受控資源的一條很好的途徑,因此我們應(yīng)該徹底地認(rèn)識它。
類層次體系中的根基類(root base class)必須實現(xiàn)IDisposable接口以釋放資源。這個類型還必須添加一個作為防御機(jī)制的終結(jié)器。所有這些程序都把釋放資源的工作委托給一個虛擬的方法,衍生的類可以根據(jù)自己的資源管理需求來重載該方法。只要衍生的類必須釋放自己的資源,并且它必須調(diào)用該函數(shù)的基類版本的時候,它才需要重載這個虛擬方法。
開始的時候,如果你的類使用了非內(nèi)存資源,它就必須含有一個終結(jié)器。你不能依賴客戶端總是調(diào)用Dispose()方法。因為當(dāng)它們忘記這樣做的時候,你就面臨資源泄漏的問題。沒有調(diào)用Dispose是它們的問題,但是你卻有過失。用于保證非內(nèi)存資源被正確地釋放的唯一途徑是建立終結(jié)器。
當(dāng)垃圾收集器運(yùn)行的時候,它立即從內(nèi)存中刪除所有不帶終結(jié)器的垃圾對象。所有帶有終結(jié)器的對象仍然存在于內(nèi)存中。這些對象都被添加到終結(jié)隊列,垃圾收集器引發(fā)一個新線程,周期性地在這些對象上運(yùn)行終結(jié)器。在這些終結(jié)程序線程完成自己的工作之后,就可以從內(nèi)存中刪除垃圾對象了。需要終結(jié)的對象在內(nèi)存中停留的時間比沒有終結(jié)器的對象停留的時間長很多。但是你別無選擇。如果要使程序有防護(hù)性,在類型包含非受控資源的時候,你必須編寫一個終結(jié)器。但是也不用擔(dān)心性能問題。下一步確保了客戶端避免與終結(jié)相關(guān)的性能開銷。
C# Dispose方法模式之實現(xiàn)IDisposable接口是一種標(biāo)準(zhǔn)的途徑,它通知用戶和運(yùn)行時系統(tǒng)持有資源的對象必須及時地釋放。IDisposable接口僅僅包含一個方法:
- public interface IDisposable
- {
- void Dispose( );
- }
C# Dispose方法模式對IDisposable.Dispose()方法的實現(xiàn)(implementation)負(fù)責(zé)下面四個事務(wù):
1、釋放所有的非受控資源。
2、釋放所有的受控資源(包括未解開事件)。
3、設(shè)置標(biāo)志表明該對象已經(jīng)被處理過了。你必須在自己的公共方法中檢查這種狀態(tài)標(biāo)志并拋出ObjectDisposed異常(如果某個對象被處理過之后再次被調(diào)用的話)。
4、禁止終結(jié)操作(finalization)。你調(diào)用GC.SuppressFinalize(this)來完成這種事務(wù)。
通過實現(xiàn)IDisposable接口你完成了兩個事務(wù):你為客戶端及時地釋放自己持有的所有受控資源提供了機(jī)制;你為客戶端提供了一種釋放非受控資源的標(biāo)準(zhǔn)途徑。這是一個很大的進(jìn)步。當(dāng)你在類型中實現(xiàn)了Idisposable接口的時候,客戶端可以避免終結(jié)操作的開銷,你的類就成為.NET世界中的"良民"了。
但是在你建立的這種機(jī)制中仍然存在一些問題。怎樣在衍生類清理自己資源的時候同時也讓基類能夠清理資源?如果衍生類重載了終結(jié)操作,或者添加了自己的IDisposable實現(xiàn),那么這些方法必須調(diào)用基類,否則,基類就不能正確地進(jìn)行清理操作。同樣,finalize(終結(jié)操作)和Dispose參與分擔(dān)了一些相同的職責(zé)。Finalize方法和Dispose方法的代碼幾乎相同。而且在重載接口函數(shù)后并不像你預(yù)料的那樣工作。標(biāo)準(zhǔn)的Dispose模式中的第三個方法是一個受保護(hù)的虛擬輔助函數(shù),它分解出這些共同的事務(wù),并給衍生類添加一個用于釋放資源的"鉤子(hook)"?;惏撕诵慕涌诘拇a。作為對Dispose()或終結(jié)操作的響應(yīng),該虛擬函數(shù)為衍生類清除資源提供了"鉤子":
- protected virtual void Dispose( bool isDisposing );
這個重載的方法實現(xiàn)支持finalize和Dispose的必要事務(wù),由于它是虛擬的,它為所有的衍生類提供了一個入口點(diǎn)。衍生類可以重載這個方法,為清除自己的資源提供適當(dāng)?shù)膶崿F(xiàn),同時還可以調(diào)用基類版本。當(dāng)isDisposing為真(true)的時候,你可以清除受控和非受控資源,當(dāng)isDisposing為假(false)的時候,你只能清除非受控資源。在這兩種情況下,你都可以調(diào)用基類的Dispose(bool)方法,讓它清除自己的資源。
C# Dispose方法模式的相關(guān)介紹就向你講解到這里,希望對你了解和學(xué)習(xí)C# Dispose方法模式有所幫助。
【編輯推薦】