国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學院 > 開發設計 > 正文

【.NET深呼吸】清理對象引用,有一個問題容易被忽略

2019-11-14 16:10:33
字體:
來源:轉載
供稿:網友

大家知道,托管代碼一個重要的特點是自動管理內存,即我們常說的垃圾回收機制,那些高大上的理論我就不重復了,有興趣的朋友可以翻書。我這個有個毛病——不喜歡很嚴肅地去說一些理論的東西,所以我不多介紹了。

一般而言,當代碼執行超出某個變量的有效范圍后,或者不再引用某個對象實例時,該實例會發生析構,垃圾回收器很可能就要清理門戶了,當然也可能不是馬上清理,也許會過一會兒再清理。

對于一些要自定義進行清理操作的類,我們會采取以下方案:

1、寫上析構函數,在析構函數中清理。

2、實現IDisposable接口,并實現Dispose方法,在方法中編寫自定義清理代碼。當該類型被實例化后,最后不再使用時會調用Dispose方法清理,如果順利清理,最后還會調用類型的析構函數。通常,如何實現了IDisposable接口,就不必再寫上析構函數了。如果希望Dispose方法被自動調用,可以在實例化對象的代碼包裝在using語句塊中,當執行完using塊時會自動調用Dispose方法。

 

可能有人笑了,老周,你太逗了,這些基礎知識誰不知道?當然,我說上面那些內容是為了繞個小圈子,以便進入主題。于是,我產生了一個疑問:是不是存在某些情景下,可能導致對象實例不會被回收呢?就算你調用了Dispose方法,就算你把變量設為null來解除引用,就算你調用GC類的方法來回收……

經過老周測試,還真有這種情況,而且很多朋友都很有可能會忽略,甚至在意識認知上誤認為對象實例已經被回收,而實際上是沒有回收的。

 

我簡單說一下這種情形:

比如有一個靜態類(靜態類的成員必是靜態的)A,里面有靜態事件。隨后在其他類的實例中處理A類的靜態事件,并且處理事件的方法就位于這個實例對象上……

不急,我們還是看真實的例子吧。假如我定義了一個靜態類MyChecker,它里面有個靜態事件CheckEvent。

    public static class MyChecker    {        #region 靜態事件        public static event EventHandler CheckEvent;        #endregion        public static void CallEvent()        {            if (CheckEvent != null)            {                CheckEvent(new object(), EventArgs.Empty);            }        }    }

只要CallEvent方法被調用,CheckEvent事件會被引發。

 

然后,定義另一個類SampleClass,并在該類中處理剛才MyChecker中的靜態事件。

    class SampleClass:IDisposable    {        public SampleClass()        {            MyChecker.CheckEvent += MyChecker_CheckEvent;        }        void MyChecker_CheckEvent(object sender, EventArgs e)        {            new Form2().Show();        }        ~SampleClass()        {            System.Diagnostics.Debug.WriteLine("/n看,析構函數調用了。/n");        }        public void Dispose()        {            //……        }    }


在類的構造函數中,附加CheckEvent事件的處理,處理方法名為MyChecker_CheckEvent。

可能大家已經發現,老周寫的SampleClass類有點恐怖氣息,既實現了Dispose方法,怎么又寫了析構函數,我這里寫上析構函數是為了驗證類的實例是否真的被清理,如果實例真的被回收,那么Debug類會在“輸出”窗口中輸出提示,如果沒有提示輸出,說明類的實例還霸占著內存。

接下來測試一下。

            SampleClass sc = new SampleClass();            await Task.Delay(10 * 1000);            sc.Dispose();            sc = null;            GC.Collect();


實例化SampleClass后,然后Delay會暫停10秒,10秒鐘過后會調用Dispose方法,并設置變量為null引用,我害怕不能及時清理,連GC.Collect方法也用上了。

而在等待這10秒期間,可以調用靜態類的CallEvent方法來引發靜態事件CheckEvent。

MyChecker.CallEvent();

按照一般理解,在10秒鐘后,SampleClass實例應該被清理,并且在“輸出”窗口會輸出提示。

好,現在試一下。

……

實驗結果表明,輸出 窗口中連鴨毛都沒有輸出,這說明10秒鐘后,SampleClass實例根本沒有發生析構。于是又出問題了,這是怎么回事?SampleClass實例不是不存在引用了嗎,怎么不發生析構?

其實我們忽略了一點:靜態事件CheckEvent還跟SampleClass實例的方法綁定著呢,實質上,雖然將變量設為null,可是SampleClass實例中的MyChecker_CheckEvent方法還被靜態類中的靜態事件引用著,所以不會被回收。不知道你明白了沒。

這個問題很多朋友在實際開發中都會忽略,還得意地以為Sample實例真的被回收了,實際上實例不會被回收,除非程序結束。因為MyChecker是靜態類,不基于實例。如果MyChecker不是靜態類,那么當MyChecker的實例釋放后,SampleClass實例就可以被釋放了。

那么,如何解決呢?很簡單,只要在SampleClass類的Dispose方法中解除靜態事件與方法的綁定即可,這樣的話,靜態事件就不再引用實例中的方法成員了,此時實例就可以發生析構了。

        public void Dispose()        {            MyChecker.CheckEvent -= MyChecker_CheckEvent;        }

 

這個例子研究,告訴我們:在類實例中處理靜態事件時一定要小心

本示例的源碼下載:http://files.VEVb.com/files/tcjiaan/refsample.zip

好了,今天就扯到這里吧。

 


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 和静县| 南漳县| 民权县| 亚东县| 万年县| 洛扎县| 海盐县| 托克逊县| 广灵县| 龙海市| 镇赉县| 涟水县| 奉化市| 和林格尔县| 庆元县| 锦屏县| 社旗县| 广平县| 田阳县| 黑河市| 黔江区| 固安县| 自贡市| 永兴县| 扎赉特旗| 平安县| 焉耆| 手游| 安庆市| 滁州市| 河曲县| 霸州市| 巴塘县| 桑植县| 蒙山县| 南京市| 冷水江市| 双牌县| 井冈山市| 莲花县| 新干县|