天天看點

C#資源釋放及Dispose、Close和析構方法

C#資源釋放及Dispose、Close和析構方法

一:什麼是資源

在開始本文前,需要一些準備知識。首先要提出“什麼是資源”。在CLR出來之後,Windows系統資源開始分為“非托管資源”和“托管資源”。

         非托管資源是指:所有的Window核心對象(句柄)都是非托管資源,如對于Stream,資料庫連接配接,GDI+的相關對象,還有Com對象等等,這些資源并不是受到CLR管理;

         托管資源是指:由CLR管理配置設定和釋放的資源,即由CLR裡new出來的對象。

其次再來講,資源的釋放方式。

         非托管資源:需要顯式釋放的,也即需要你寫代碼釋放;

         托管資源:并不需要顯式釋放,但是如果引用類型本身含有非托管資源,則需要進行現實釋放;

二:顯式釋放的C#實作

顯式釋放的C#實作,由C#文法支援的有:

         1:實作IDisposable接口的Dispose方法;

         2:析構方法(終結器);

         不由C#文法支援,但是約定支援的顯式釋放是:

         3:提供顯示釋放方法,比如常用的Close方法;

三:Dispose、Close和析構方法異同點

但是,還需要區分這3種方式的異同點。首先,你無法調用析構方法。析構方法是由垃圾回收機制進行調用的。換句話來說,就是你不知道析構方法被調用的時機。嚴格意義上來說,它隻是作為資源釋放的一個補救措施。

資源釋放的一個正确的措施是為類型實作IDisposable接口的Dispose。當你需要釋放類型的資源的時候,應該顯示的調用Dipose方法。當然,這裡還有一個C#的文法糖,就是使用using程式塊,在離開using程式塊的時候,CLR會自動調用類型所建立對象的Dipose方法。

可能有人會問道,既然可以通過Dispose方法的方式來進行資源的釋放,為什麼有些類型還需要提供一個Close方法。這裡面的差別,或者說約定在于,如果你仔細觀察這些類型:他們基本都隻公開了Close方法,他們都實作了IDisposable,但都隐藏了Dispose方法。以Socket這個類為例,它:

1:提供public void Close()

public void Close()

{

//….

((IDisposable)this).Dispose();

}

2:提供顯式void IDisposable.Dispose()

void IDisposable.Dispose()

this.Dispose(true);

GC.SuppressFinalize(this);

3:提供protected virtual void Dispose(bool disposing)。真正的資源釋放的代碼放在這裡。

是以理論上來将,提供Close方法最終還是使用的Dispose方法,之是以這麼做,是因為這些類型出于顯式實作IDisposable的因素,在調用這些Dispose方法的時候,必須完成一次轉型,如: 

           ((IDisposable)new A()).Dispose(); 

為了避免轉型,同時也為了避免不熟悉C#文法的開發人員更直覺的釋放資源,提供了Close方法。

在上文的例子中,你可能已經注意到IDisposable.Dispose這個方法中,包含一句: 

      GC.SuppressFinalize(this); 

這是告訴CLR,在進行垃圾回收的時候,不用再繼續調用析構方法(終結器)了。是的,因為你已經手動釋放資源了。這也從另一個方面驗證了析構方法隻是作為資源釋放的補救機制。因為假設你忘記Close或者Dispose了,CLR會在垃圾回收的時候為你做這件事。檢視Socket的析構函數,你會很好的了解這一點。

~Socket()

this.Dispose(false);

是的,析構方法調用的也是Dispose。

備注1:本文帶來幾個争論

1:托管資源本身是否需要顯式釋放。答案顯然是:不需要;

2:如果引用類型對象不再需要,是否需要顯式=null;答案是:即使不這樣做,GC也會進行垃圾回收。

3:将托管資源分為引用類型資源和值類型資源這種分類方法是有問題的,或者說是錯誤的。正确的分類法應該是棧資源和堆資源。線程棧中存放的是方法的實參和方法内部的局部變量。堆上存放的是類型對象本身及對象的兩個額外成員:類型對象指針和同步塊索引。

4:Dispose方法本身是用來讓你放置資源清理代碼的。顯然,一個空方法并不代表清理工作本身,真正執行清理工作的是你具體的代碼。

備注2:推薦Dipose模式實作

如:基類

C#資源釋放及Dispose、Close和析構方法
C#資源釋放及Dispose、Close和析構方法

代碼

class ClassShouldDisposeBase : IDisposable

public void Dispose()

protected virtual void Dispose(bool disposing)

if (disposing)

//執行基本的清理代碼

~ClassShouldDisposeBase()

C#資源釋放及Dispose、Close和析構方法

子類:

C#資源釋放及Dispose、Close和析構方法
C#資源釋放及Dispose、Close和析構方法

class ClassShouldDispose: ClassShouldDisposeBase

// 執行子類清理代碼

// 如有必要,執行base.Dispose(disposing);

else

//調用本類或者基類的Dispose方法

//其它代碼

C#資源釋放及Dispose、Close和析構方法

本文轉自最課程陸敏技部落格園部落格,原文連結:http://www.cnblogs.com/luminji/archive/2011/01/05/1926468.html,如需轉載請自行聯系原作者