a亚洲精品_精品国产91乱码一区二区三区_亚洲精品在线免费观看视频_欧美日韩亚洲国产综合_久久久久久久久久久成人_在线区

首頁(yè) > 編程 > C# > 正文

詳解C#編程中.NET的弱事件模式

2019-10-29 21:41:14
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

這篇文章主要介紹了C#編程中.NET的弱事件模式,深入討論了C#中相關(guān)的垃圾回收機(jī)制,需要的朋友可以參考下

引言

你可能知道,事件處理是內(nèi)存泄漏的一個(gè)常見來(lái)源,它由不再使用的對(duì)象存留產(chǎn)生,你也許認(rèn)為它們應(yīng)該已經(jīng)被回收了,但不是,并有充分的理由。

在這個(gè)短文中(期望如此),我會(huì)在 .Net 框架的上下文事件處理中展示這個(gè)問題,之后我會(huì)教你這個(gè)問題的標(biāo)準(zhǔn)解決方案,弱事件模式。有兩種方法,即:

“傳統(tǒng)”方法 (嗯,在 .Net 4.5 前,所以也沒那么老),它實(shí)現(xiàn)起來(lái)比較繁瑣

.Net 4.5 框架提供的新方法,它則是盡其可能的簡(jiǎn)單

(源代碼在這里可供使用。)

從常見事物開始

在一頭扎進(jìn)本文核心內(nèi)容前,讓我們回顧一下在代碼中最常使用的兩個(gè)事物:類和方法。

事件源

讓我為您介紹一個(gè)基本但很有用的事件源類,它最低限度地揭示了足夠的復(fù)雜性來(lái)說(shuō)明這一點(diǎn):

 

 
  1. public class EventSource 
  2. public event EventHandlerEvent = delegate { }; 
  3.  
  4. public void Raise() 
  5. Event(this, EventArgs.Empty); 

對(duì)好奇那個(gè)奇怪的空委托初始化方法(delegate { })的人來(lái)說(shuō),這是一個(gè)用來(lái)確保事件總被初始化的技巧,這樣就可以不必每次在使用它之前都要檢查它是否不為NULL。

觸發(fā)垃圾收集的實(shí)用方法

在.net中,垃圾收集以一種不確定的方式觸發(fā)。這對(duì)我們的實(shí)驗(yàn)很不利,我們的實(shí)驗(yàn)需要以一種確定的方式跟蹤對(duì)象的狀態(tài)。

所以,我們必須定期觸發(fā)自己的垃圾收集操作,同時(shí)避免復(fù)制管道代碼,管道代碼已經(jīng)在在一個(gè)特定的方法中釋放:

 

 
  1. static void TriggerGC() 
  2. Console.WriteLine("Starting GC."); 
  3.  
  4. GC.Collect(); 
  5. GC.WaitForPendingFinalizers(); 
  6. GC.Collect(); 
  7.  
  8. Console.WriteLine("GC finished."); 

雖然不是很復(fù)雜,但是如果你不是很熟悉這種模式,還是有必要小小解釋一下:

第一個(gè) GC.Collect() 觸發(fā).net的CLR垃圾收集器,對(duì)于負(fù)責(zé)清理不再使用的對(duì)象,和那些類中沒有終結(jié)器(即c#中的析構(gòu)函數(shù))的對(duì)象,CLR垃圾收集器足夠勝任

GC.WaitForPendingFinalizers() 等待其他對(duì)象的終結(jié)器執(zhí)行;我們需要這樣做,因?yàn)椋銓⒖吹轿覀兪褂媒K結(jié)器方法去追蹤我們的對(duì)象在什么時(shí)候被收集的

第二個(gè)GC.Collect() 確保新生成的對(duì)象也被清理了

引入問題

首先讓我們?cè)囍ㄟ^一些理論,最重要的是還有一個(gè)演示的幫助,去了解事件監(jiān)聽器有哪些問題。

背景

一個(gè)對(duì)象要想被作為事件偵聽器,需要將其實(shí)例方法之一登記為另一個(gè)能夠產(chǎn)生事件的對(duì)象(即事件源)的事件處理程序,事件源必須保持一個(gè)到事件偵聽器對(duì)象的引用,以便在事件發(fā)生時(shí)調(diào)用此偵聽器的處理方法。

這很合理,但如果這個(gè)引用是一個(gè) 強(qiáng)引用,則偵聽器會(huì)作為事件源的一個(gè)依賴 從而不能作為垃圾回收,即使引用它的最后一個(gè)對(duì)象是事件源。

下面詳細(xì)圖解在這下面發(fā)生了什么:

事件處理問題

這將不是一個(gè)問題,如果你可以控制listener object的生命周期,你可以取消對(duì)事件源的訂閱當(dāng)當(dāng)你不再需要listener,常常可以使用disposable pattern(用后就扔的模式)。

但是如果你不能在listener生命周期內(nèi)驗(yàn)證單點(diǎn)響應(yīng),在確定性的方式中你不能把它處理掉,你必須依賴GC處理...這將從不會(huì)考慮你所準(zhǔn)備的對(duì)象,只要事件源還存在著!

例子

理論都是好的,但還是讓我們看看問題和真正的代碼。

這是我們勇敢的時(shí)間監(jiān)聽器,還有點(diǎn)幼稚,我們很快知道為什么:

 

 
  1. public class NaiveEventListener 
  2. private void OnEvent(object source, EventArgs args) 
  3. Console.WriteLine("EventListener received event."); 
  4.  
  5. public NaiveEventListener(EventSource source) 
  6. source.Event += OnEvent; 
  7.  
  8. ~NaiveEventListener() 
  9. Console.WriteLine("NaiveEventListener finalized."); 

用一個(gè)簡(jiǎn)單例子來(lái)看看怎么實(shí)現(xiàn)運(yùn)作:

 

 
  1. Console.WriteLine("=== Naive listener (bad) ==="); 
  2.  
  3. EventSource source = new EventSource(); 
  4.  
  5. NaiveEventListener listener = new NaiveEventListener(source); 
  6.  
  7. source.Raise(); 
  8.  
  9. Console.WriteLine("Setting listener to null."); 
  10. listener = null
  11.  
  12. TriggerGC(); 
  13.  
  14. source.Raise(); 
  15.  
  16. Console.WriteLine("Setting source to null."); 
  17. source = null
  18.  
  19. TriggerGC(); 

輸出:

 

 
  1. EventListener received event. 
  2. Setting listener to null
  3. Starting GC. 
  4. GC finished. 
  5. EventListener received event. 
  6. Setting source to null
  7. Starting GC. 
  8. NaiveEventListener finalized. 
  9. GC finished. 

讓我們分析下這個(gè)運(yùn)作流程:

“EventListener received event.“:這是我們調(diào)用 “source.Raise()”的結(jié)果; perfect, seems like we're listening.

“Setting listener to null.“: 我們把本地事件監(jiān)聽器對(duì)象引用賦空值,這樣應(yīng)該可以讓垃圾回收器回收了.

“Starting GC.“: 垃圾回收開始.

“GC finished.“: 垃圾回收開始, 但是 但是我們的事件監(jiān)聽器沒有被回收器回收, 這樣就證明了事件監(jiān)聽器的析構(gòu)函數(shù)沒有被調(diào)用。

“EventListener received event.“: 第二次調(diào)用 “source.Raise()”來(lái)確認(rèn),發(fā)現(xiàn)這監(jiān)聽器還活著。

“Setting source to null.“: 我們?cè)谫x空值給事件的原對(duì)象.

“Starting GC.“: 第二次垃圾回收.

“NaiveEventListener finalized.“: 這一次幼稚的事件監(jiān)聽終于被回收了,遲到總好過沒有.

“GC finished.“:第二次垃圾回收完成.

結(jié)論:確實(shí)有一個(gè)隱藏的對(duì)事件監(jiān)聽器的強(qiáng)引用,目的是防止它在事件源被回收之前被回收!

希望有針對(duì)此問題的標(biāo)準(zhǔn)解決方案:讓事件源可以通過弱引用來(lái)引用偵聽器,在事件源存在時(shí)也可以回收偵聽器對(duì)象。

這里有一個(gè)標(biāo)準(zhǔn)的模式及其在.NET框架上的實(shí)現(xiàn):弱事件模式(http://msdn.microsoft.com/en-us/library/aa970850.aspx)。 And there is a standard pattern and its implementation in the .Net framework: the weak event pattern.

弱事件模式

讓我們看看在.NET中如何應(yīng)付這個(gè)問題,

通常有超過一種方法去做,但是在這種情況下可以直接決定:

如果你正在使用 .Net 4.5 ,那么你將從簡(jiǎn)單的實(shí)現(xiàn)受益

另外,你必須依靠一點(diǎn)人為的技巧手段

傳統(tǒng)方式

WeakEventManager 是所有模式管道的封裝

IWeakEventListener 是管道,它允許一個(gè)組件連接到WeakEventManager管件

(這兩個(gè)位于WindowBase程序集,你將需要參考你自己的如果你不在開發(fā)WPF項(xiàng)目,你應(yīng)該準(zhǔn)確的參考WindowBase)

因此這有兩步處理.

首先通過繼承WeakEventManager來(lái)實(shí)現(xiàn)一個(gè)自定義事件管理器:

重寫 StartListening 和 StopListening 方法,分別注冊(cè)一個(gè)新的handler和注銷一個(gè)已存在的; 它們將被WeakEventManager基類使用。

提供兩個(gè)方法來(lái)訪問listener列表, 命名為 “AddListener” 和 “RemoveListener “,給自定義事件管理器的使用者使用。

通過在自定義事件管理器上暴露一個(gè)靜態(tài)屬性,提供一個(gè)方式去獲得當(dāng)前線程的事件管理器。

之后使listenr實(shí)現(xiàn)IWeakEventListenr接口:

實(shí)現(xiàn) ReceiveWeakEvent 方法

嘗試去處理這個(gè)事件

如果無(wú)誤的處理好事件,將返回true

有很多要說(shuō)的,但是可以相對(duì)地轉(zhuǎn)換成一些代碼:

首先是自定義弱事件管理器:

 

 
  1. public class EventManager : WeakEventManager 
  2. private static EventManager CurrentManager 
  3. get 
  4. EventManager manager = (EventManager)GetCurrentManager(typeof(EventManager)); 
  5.  
  6. if (manager == null
  7. manager = new EventManager(); 
  8. SetCurrentManager(typeof(EventManager), manager); 
  9.  
  10. return manager; 
  11.  
  12.  
  13. public static void AddListener(EventSource source, IWeakEventListener listener) 
  14. CurrentManager.ProtectedAddListener(source, listener); 
  15.  
  16. public static void RemoveListener(EventSource source, IWeakEventListener listener) 
  17. CurrentManager.ProtectedRemoveListener(source, listener); 
  18.  
  19. protected override void StartListening(object source) 
  20. ((EventSource)source).Event += DeliverEvent; 
  21.  
  22. protected override void StopListening(object source) 
  23. ((EventSource)source).Event -= DeliverEvent; 

之后是事件listener:

 

 
  1. public class LegacyWeakEventListener : IWeakEventListener 
  2. private void OnEvent(object source, EventArgs args) 
  3. Console.WriteLine("LegacyWeakEventListener received event."); 
  4.  
  5. public LegacyWeakEventListener(EventSource source) 
  6. EventManager.AddListener(source, this); 
  7.  
  8. public bool ReceiveWeakEvent(Type managerType, object sender, EventArgs e) 
  9. OnEvent(sender, e); 
  10.  
  11. return true
  12.  
  13. ~LegacyWeakEventListener() 
  14. Console.WriteLine("LegacyWeakEventListener finalized."); 

檢查下:

 

  1. Console.WriteLine("=== Legacy weak listener (better) ==="); 
  2.  
  3. EventSource source = new EventSource(); 
  4.  
  5. LegacyWeakEventListener listener = new LegacyWeakEventListener(source); 
  6.  
  7. source.Raise(); 
  8.  
  9. Console.WriteLine("Setting listener to null."); 
  10. listener = null
  11.  
  12. TriggerGC(); 
  13.  
  14. source.Raise(); 
  15.  
  16. Console.WriteLine("Setting source to null."); 
  17. source = null
  18.  
  19. TriggerGC(); 

輸出:

 

 
  1. LegacyWeakEventListener received event. 
  2. Setting listener to null
  3. Starting GC. 
  4. LegacyWeakEventListener finalized. 
  5. GC finished. 
  6. Setting source to null
  7. Starting GC. 
  8. GC finished. 

非常好,它起作用了,我們的事件listener對(duì)象現(xiàn)在可以在第一次GC里正確的析構(gòu),即使事件源對(duì)象還存活,不再泄露內(nèi)存了.

但是要寫一堆代碼就為了一個(gè)簡(jiǎn)單的listener,想象一下你有一堆這樣的listener,你必須要為每個(gè)類型的寫一個(gè)弱事件管理器!

如果你很擅長(zhǎng)代碼重構(gòu),你可以發(fā)現(xiàn)一個(gè)聰明的方式去重構(gòu)所有通用的代碼.

在.Net 4.5 出現(xiàn)之前,你必須自己實(shí)現(xiàn)弱事件管理器,但是現(xiàn)在,.Net提供一個(gè)標(biāo)準(zhǔn)的解決方案來(lái)解決這個(gè)問題了,現(xiàn)在就來(lái)回顧下吧!

.Net 4.5 方式

.Net 4.5 已介紹了一個(gè)新的泛型版本的遺留WeakEventManager: WeakEventManager.

(這個(gè)類可以在WindowsBase集合.)

多虧了 .Net WeakEventManager 自己處理泛型, 不用去一個(gè)個(gè)實(shí)現(xiàn)新事件管理器.

而且代碼還簡(jiǎn)單和可讀:

 

 
  1. public class WeakEventListener 
  2. private void OnEvent(object source, EventArgs args) 
  3. Console.WriteLine("WeakEventListener received event."); 
  4.  
  5. public WeakEventListener(EventSource source) 
  6. WeakEventManager.AddHandler(source, "Event", OnEvent); 
  7.  
  8. ~WeakEventListener() 
  9. Console.WriteLine("WeakEventListener finalized."); 

簡(jiǎn)單的一行代碼,真簡(jiǎn)潔.

其他實(shí)現(xiàn)的使用也是相似的, 就是裝入所有東西到事件listener類里:

 

 
  1. Console.WriteLine("=== .Net 4.5 weak listener (best) ==="); 
  2.  
  3. EventSource source = new EventSource(); 
  4.  
  5. WeakEventListener listener = new WeakEventListener(source); 
  6.  
  7. source.Raise(); 
  8.  
  9. Console.WriteLine("Setting listener to null."); 
  10. listener = null
  11.  
  12. TriggerGC(); 
  13.  
  14. source.Raise(); 
  15.  
  16. Console.WriteLine("Setting source to null."); 
  17. source = null
  18.  
  19. TriggerGC(); 

輸出也是肯定正確的:

 

 
  1. WeakEventListener received event. 
  2. Setting listener to null
  3. Starting GC. 
  4. WeakEventListener finalized. 
  5. GC finished. 
  6. Setting source to null
  7. Starting GC. 
  8. GC finished. 

預(yù)期結(jié)果也跟之前一樣,還有什么問題?!

結(jié)論

正如你看到的,在.Net上實(shí)現(xiàn)弱事件模式 是十分直接, 特別在 .Net 4.5.

如果你沒有用.Net 4.5來(lái)實(shí)現(xiàn),將需要一堆代碼, 你可能不去用任何模式而是直接使用C# (+= and -=), 看看是否有內(nèi)存問題,如果注意到泄露,還需要花必要的時(shí)間去實(shí)現(xiàn)一個(gè)。

但是用 .Net 4.5, 它是自由和簡(jiǎn)潔,而且由框架管理, 你可以毫無(wú)顧慮的選擇它, 盡管沒有 C# 語(yǔ)法 “+=” 和 “-=” 的酷, 但是語(yǔ)義是清晰的,這才是最重要的.

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 国产不卡一区二区三区在线观看 | 久久久久久一区 | 欧美成人一区二区三区片免费 | 精品久久久久久久久久久久久久 | 国产在线中文字幕 | 天天干天天插 | www.日本精品 | 国产高清视频 | 正在播放国产精品 | 四虎免费影视 | 久久精品欧美一区二区三区不卡 | 一级黄色a视频 | 日韩在线一 | 精品国产一区二区在线 | 欧美一级欧美三级在线观看 | 国产精品.xx视频.xxtv | 亚洲一区二区三 | 天天干天天操 | a√免费视频| 一区二区三区四区在线 | 懂色av中文一区二区三区天美 | 中文字幕在线观看亚洲 | 亚洲一区二区三区在线播放 | 最新国产在线视频 | 欧美日影院| 国产免费av在线 | 中文字幕电影在线 | 国内久久精品视频 | 欧美一级片 | 久艹在线视频 | 九九九九精品九九九九 | 午夜在线视频 | 中文精品久久久 | 午夜精品久久久久久久白皮肤 | 爱爱视频在线免费观看 | 日本一区二区精品 | 国产一级中文字幕 | 国产91一区 | 黄色免费观看 | 欧美国产日韩在线 | 黄色影视免费观看 |