C#基礎:Dispose方法和Finalize方法在何時被調用


一、前言

在C#中,由於有了垃圾回收機制的支持,對象的析構和以前的C++有了很大的不同,這就要求程序員在設計類型的時候,充分理解.NET的機制,明確怎樣利用Dispose方法和Finalize方法來保證一個對象正確而高效地被析構。

二、Dispose方法的功能

我們在講解有關using的用法時,已經介紹了Dispose方法。正是因為垃圾回收機制掩蓋了對象內存真正被回收的時間,考慮到很多情況下程序員扔希望在對象不再被使用的時候進行一些清理工作,所以.NET提供了IDisposable接口並且在其中定義了Dispose方法。通常程序員會在Dispose方法中實現一些托管對象和非托管對象的釋放以及邏輯業務的結束工作等。注意實現了Dispose方法不能得到任何有關釋放的保證,Dispose方法的調用依賴於類型的使用者,當類型被不恰當地使用時,Dispose方法將不會被調用,但using等語法的存在還是幫助了類型的Dispose方法被調用。

三、Finalize方法的機制

由於Dispose方法的調用依賴於使用者,為了彌補這一缺陷,.NET同時提供了Finalize方法。Finalize方法常常被具有C++開發經驗的程序員稱為析構方法,但它的執行方法卻和傳統C++中的析構函數完全不同。Finalize方法在GC執行垃圾回收時調用,具體的機制是這樣的:

  • 當每個包含Finalize方法的類型的實例對象被分配時,.NET會在一張特定的表結構中添加一個引用並且執行這個實例對象。方便起見稱該表為“帶析構對象表”。
  • 當GC執行並且檢測到一個不被使用的對象時,需要進一步檢查“帶析構對象表”來查看該對象類型是否有Finalize方法,如果沒有則該對象被視為垃圾,如果存在Finalize方法,則把指向該對象的引用從“帶析構對象表”移到另外一張表中,這里暫時稱它為“等待析構表”。並且該對象實例被視為扔在被使用。
  • CLR將有一個單獨的線程負責處理“等待析構表”,其方法就是依次通過引用調用其中每個對象的Finalize方法,然后刪除引用,這時托管堆中的對象實例將處於不再被使用的狀態。
  • 下一個GC執行時,將釋放已經被調用Finalize方法的那些對象實例。

四、正確地使用Dispose和Finalize方法

Finalize方法確實比Dispose方法更加安全,因為它由CLR保證調用,但是性能方面Finalize方法卻要差的多。我們需要知道的是:正確的類型設計是把Finalize方法作為Dispose方法的后備,只有在使用者沒有調用Dispose方法的情況下,Finalize方法才能被視為需要執行。下面是一個正確高效的設計模板,建議牢記這個模板並且套用到每一個需要DIspose和Finalize方法的類型上去。

using System;

namespace usingDemo
{
    public class FinalizeDisposeBase : IDisposable
    {
        // 標記對象是否已被釋放
        private bool _disposed = false;
        // Finalize方法
        ~FinalizeDisposeBase()
        {
            Dispose(false);
        }

        /// <summary>
        /// 這里實現了IDisposable中的Dispose方法
        /// </summary>
        public void Dispose()
        {
            Dispose(true);
            // 告訴GC此對象的Finalize方法不再需要調用
            GC.SuppressFinalize(true);
        }

        /// <summary>
        /// 在這里做實際的析構工作
        /// 聲明為虛方法以供子類在必要時重寫
        /// </summary>
        /// <param name="isDisposing"></param>
        protected virtual void Dispose(bool isDisposing)
        {
            // 當對象已經被析構時,不在執行
            if(_disposed)
            {
                return;
            }
            if(isDisposing)
            {
                // 在這里釋放托管資源
                // 只在用戶調用Dispose方法時執行
            }
            // 在這里釋放非托管資源
            // 標記對象已被釋放
            _disposed = true;
        }
    }

    public sealed class FinalizeDispose:FinalizeDisposeBase
    {
        private bool _mydisposed = false;
        protected override void Dispose(bool isDisposing)
        {
            // 保證只釋放一次
            if (_mydisposed)
            {
                return;
            }
            if(isDisposing)
            {
                // 在這里釋放托管的並且在這個類型中聲明的資源
            }
            // 在這里釋放非托管的並且在這個類型中聲明的資源
            // 調用父類的Dispose方法來釋放父類中的資源
            base.Dispose(isDisposing);
            // 設置子類的標記
            _mydisposed = true;
        }
        static void Main()
        {

        }
    }
}

上面的代碼是一個近乎完美的Dispose配合Finalize的設計模板,其中有幾點需要特別注意:

  • 真正做釋放工作的只是Virtual的受保護方法Dispose方法,事實上這個方法的名字並不重要,僅僅為了通用和更好理解,稱呼它為Dispose。
  • 虛方法Dispose需要接受一個布爾類型的參數,主要用於區分調用方是類型的使用者還是.NET的垃圾回收。前者通過IDisposable的Dispose方法,而后者通過Finalize方法。兩者的區別是通過Finalize方法釋放資源時不能再釋放或使用對象中的托管資源,這是因為這時的對象已經處於不被使用的狀態,很有可能其中的托管資源已經被釋放掉了。
  • 在IDisposable的Dispose方法的實現中通過GC.SuppressFinalize()方法來告訴.NET此對象在被回收時不需要調用Finalize方法,這一句是改善性能的關鍵,記住實現Dispose方法的本質目的就是避免所有釋放工作在Finalize方法中進行。
  • 子類型必須定義自己的釋放標記來標明子類中的資源是否已經被釋放,同時子類的虛方法Dispose方法也只需要釋放自己新定義的資源。
  • 確保在虛方法Dispose中做的都是釋放工作,有些邏輯上的結束工作需要反復斟酌,以防止一個簡單的賦值語句使對象再度存活。

五、總結

Dispose方法被使用者主動調用,而Finalize方法在對象被垃圾回收的第一輪回收后,由一個專用的.NET線程進行調用。Dispose方法不能保證被執行,而.NET的垃圾回收機制保證了擁有Finalize方法並且需要被調用的類型對象的Finalize方法被執行。調用Finalize方法涉及了一系列復雜的操作,性能代價非常高,程序員可以通過GC.SuppressFinalize方法通知.NET該對象的Finalize方法不需要被調用。有關Dispose和Finalize的類型設計應該參照上面的代碼模板,以保證對象能夠被高效和安全的釋放。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM