解構函式
解構函式不能有修飾符,如public。不能接受任何參數。
編譯器自動將一個解構函式轉換成對Object.Finalize方法的一個override版,如下。
class Test{ protected override void Finalize() { try {…} finally { base.Finalize(); } }}
記憶體回收行程
.NET記憶體回收行程會保證:
l 每個對象都會被摧毀,它的解構函式一定會被運行。當一個程式結束後,所有對象都會被銷毀。
l 每個對象只被銷毀一次。
l 每個對象只有在不可抵達時(即不存在到該對象的引用時)才會被銷毀。
工作方式:
1) 它構造包含所有可抵達對象的一個map。為此,它會反覆跟隨對象中的引用欄位。記憶體回收行程會非常小心地構造這個map,並確保循環參考不會無限遞迴。這個map中任意對象都不會被認為不可抵達。
2) 它檢查是否有任何不可抵達的對象具有一個需要啟動並執行解構函式(運行解構函式的過程稱為finalization)。需要finalization的任何不可抵達的對象都會放到一個特殊的隊列中。這個隊列稱為freachable隊列。
3) 它回收剩餘的不可抵達的對象(即不需要finalization的對象)。為此,它會在heap中向下移動可抵達的對象,從而對heap進行磁碟重組,並釋放位於heap頂部的記憶體。當記憶體回收行程移動一個可抵達的對象時,還會更新對該對象的引用。
4) 然後,它允許其他線程恢複執行
5) 它在一個單獨的線程中,對需要finalization的不可抵達的對象(位於freachable隊列中)執行finalize操作。
有上面的總結可以看出,解構函式的存在會使上面的過程多執行2,5兩步。所以考慮使用using塊來代替泛型。如果所使用的一個類實現了Dispose方法(Close方法)。最好是在finally中調用這個方法(調用方法前需檢查這個要被dispose的對象的disposed屬性是否為false,只有在不為true時再dispose,這也是推薦使用using的原因,using很容易約束這個待析構的變數的範圍-即一對大括弧之間)。或者使用using塊將使用這個類的程式碼封裝圍。放入using塊的對象的類型必須實現IDisposable介面。
標準清理模式
最後給出一個.NET推薦使用的標準清理模式代碼,範例程式碼:
class MyClass : IDisposable{ private bool disposed = false;//Disposal 狀態 public void Dispose()//公有Dispose方法(可選實現IDisposal介面) { Dispose(true); GC.SuppressFinalize(this); } ~MyClass() { Dispose(false); } protected virtual void Dispose(bool disposing) { if (!disposed) { if (disposing) { //Dispose the managed resources. } //Dispose the unmanaged resources. } disposed = true; }}
上述代碼中,我們從解構函式中調用Dispose方法,這樣可以確保Dispose執行。,另外GC.SuppressFinalize(this);用於阻止編譯器在這個對象上執行析構。
感謝閱讀,希望能協助到大家,謝謝大家對topic.alibabacloud.com的支援!