欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時間:2015年07月13日 09:29:27   投稿:goldensun  
這篇文章主要介紹了C#編程中.NET的弱事件模式,深入討論了C#中相關(guān)的垃圾回收機制,需要的朋友可以參考下

 引言

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

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

  •     “傳統(tǒng)”方法 (嗯,在 .Net 4.5 前,所以也沒那么老),它實現(xiàn)起來比較繁瑣
  •     .Net 4.5 框架提供的新方法,它則是盡其可能的簡單

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

從常見事物開始

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

讓我為您介紹一個基本但很有用的事件源類,它最低限度地揭示了足夠的復雜性來說明這一點:

 

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

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

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

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

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

static void TriggerGC()
{
  Console.WriteLine("Starting GC.");
 
  GC.Collect();
  GC.WaitForPendingFinalizers();
  GC.Collect();
 
  Console.WriteLine("GC finished.");
}

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

  •     第一個 GC.Collect() 觸發(fā).net的CLR垃圾收集器,對于負責清理不再使用的對象,和那些類中沒有終結(jié)器(即c#中的析構(gòu)函數(shù))的對象,CLR垃圾收集器足夠勝任
  •     GC.WaitForPendingFinalizers() 等待其他對象的終結(jié)器執(zhí)行;我們需要這樣做,因為,你將看到我們使用終結(jié)器方法去追蹤我們的對象在什么時候被收集的
  •     第二個GC.Collect() 確保新生成的對象也被清理了

引入問題

首先讓我們試著通過一些理論,最重要的是還有一個演示的幫助,去了解事件監(jiān)聽器有哪些問題。
背景

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

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


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

事件處理問題

這將不是一個問題,如果你可以控制listener object的生命周期,你可以取消對事件源的訂閱當當你不再需要listener,常??梢允褂胐isposable pattern(用后就扔的模式)。

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

例子

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

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

public class NaiveEventListener
{
  private void OnEvent(object source, EventArgs args)
  {
    Console.WriteLine("EventListener received event.");
  }
 
  public NaiveEventListener(EventSource source)
  {
    source.Event += OnEvent;
  }
 
  ~NaiveEventListener()
  {
    Console.WriteLine("NaiveEventListener finalized.");
  }
}

用一個簡單例子來看看怎么實現(xiàn)運作:
 

Console.WriteLine("=== Naive listener (bad) ===");
 
EventSource source = new EventSource();
 
NaiveEventListener listener = new NaiveEventListener(source);
 
source.Raise();
 
Console.WriteLine("Setting listener to null.");
listener = null;
 
TriggerGC();
 
source.Raise();
 
Console.WriteLine("Setting source to null.");
source = null;
 
TriggerGC();

輸出:

 

EventListener received event.
Setting listener to null.
Starting GC.
GC finished.
EventListener received event.
Setting source to null.
Starting GC.
NaiveEventListener finalized.
GC finished.

讓我們分析下這個運作流程:

  •     “EventListener received event.“:這是我們調(diào)用 “source.Raise()”的結(jié)果; perfect, seems like we're listening.
  •     “Setting listener to null.“: 我們把本地事件監(jiān)聽器對象引用賦空值,這樣應(yīng)該可以讓垃圾回收器回收了.
  •     “Starting GC.“: 垃圾回收開始.
  •     “GC finished.“: 垃圾回收開始, 但是 但是我們的事件監(jiān)聽器沒有被回收器回收, 這樣就證明了事件監(jiān)聽器的析構(gòu)函數(shù)沒有被調(diào)用。
  •     “EventListener received event.“: 第二次調(diào)用 “source.Raise()”來確認,發(fā)現(xiàn)這監(jiān)聽器還活著。
  •     “Setting source to null.“: 我們在賦空值給事件的原對象.
  •     “Starting GC.“: 第二次垃圾回收.
  •     “NaiveEventListener finalized.“: 這一次幼稚的事件監(jiān)聽終于被回收了,遲到總好過沒有.
  •     “GC finished.“:第二次垃圾回收完成.


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

希望有針對此問題的標準解決方案:讓事件源可以通過弱引用來引用偵聽器,在事件源存在時也可以回收偵聽器對象。

這里有一個標準的模式及其在.NET框架上的實現(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)付這個問題,

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

  •     如果你正在使用 .Net 4.5 ,那么你將從簡單的實現(xiàn)受益
  •     另外,你必須依靠一點人為的技巧手段

傳統(tǒng)方式

  •     WeakEventManager 是所有模式管道的封裝
  •     IWeakEventListener 是管道,它允許一個組件連接到WeakEventManager管件

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


因此這有兩步處理.

首先通過繼承WeakEventManager來實現(xiàn)一個自定義事件管理器:

  •     重寫 StartListening 和 StopListening 方法,分別注冊一個新的handler和注銷一個已存在的; 它們將被WeakEventManager基類使用。
  •     提供兩個方法來訪問listener列表, 命名為 “AddListener” 和 “RemoveListener “,給自定義事件管理器的使用者使用。
  •     通過在自定義事件管理器上暴露一個靜態(tài)屬性,提供一個方式去獲得當前線程的事件管理器。
  • 之后使listenr實現(xiàn)IWeakEventListenr接口:
  •     實現(xiàn) ReceiveWeakEvent 方法
  •     嘗試去處理這個事件
  •     如果無誤的處理好事件,將返回true


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

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

public class EventManager : WeakEventManager
{
  private static EventManager CurrentManager
  {
    get
    {
      EventManager manager = (EventManager)GetCurrentManager(typeof(EventManager));
 
      if (manager == null)
      {
        manager = new EventManager();
        SetCurrentManager(typeof(EventManager), manager);
      }
 
      return manager;
    }
  }
 
 
  public static void AddListener(EventSource source, IWeakEventListener listener)
  {
    CurrentManager.ProtectedAddListener(source, listener);
  }
 
  public static void RemoveListener(EventSource source, IWeakEventListener listener)
  {
    CurrentManager.ProtectedRemoveListener(source, listener);
  }
 
  protected override void StartListening(object source)
  {
    ((EventSource)source).Event += DeliverEvent;
  }
 
  protected override void StopListening(object source)
  {
    ((EventSource)source).Event -= DeliverEvent;
  }
}

之后是事件listener:
 

public class LegacyWeakEventListener : IWeakEventListener
{
  private void OnEvent(object source, EventArgs args)
  {
    Console.WriteLine("LegacyWeakEventListener received event.");
  }
 
  public LegacyWeakEventListener(EventSource source)
  {
    EventManager.AddListener(source, this);
  }
 
  public bool ReceiveWeakEvent(Type managerType, object sender, EventArgs e)
  {
    OnEvent(sender, e);
 
    return true;
  }
 
  ~LegacyWeakEventListener()
  {
    Console.WriteLine("LegacyWeakEventListener finalized.");
  }
}

檢查下:
 

Console.WriteLine("=== Legacy weak listener (better) ===");
 
EventSource source = new EventSource();
 
LegacyWeakEventListener listener = new LegacyWeakEventListener(source);
 
source.Raise();
 
Console.WriteLine("Setting listener to null.");
listener = null;
 
TriggerGC();
 
source.Raise();
 
Console.WriteLine("Setting source to null.");
source = null;
 
TriggerGC();

輸出:
 

LegacyWeakEventListener received event.
Setting listener to null.
Starting GC.
LegacyWeakEventListener finalized.
GC finished.
Setting source to null.
Starting GC.
GC finished.

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

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

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

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

 .Net 4.5 方式

.Net 4.5 已介紹了一個新的泛型版本的遺留WeakEventManager: WeakEventManager<TEventSource, TEventArgs>.

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

多虧了 .Net WeakEventManager<TEventSource, TEventArgs> 自己處理泛型, 不用去一個個實現(xiàn)新事件管理器.

而且代碼還簡單和可讀:
 

public class WeakEventListener
{
  private void OnEvent(object source, EventArgs args)
  {
    Console.WriteLine("WeakEventListener received event.");
  }
 
  public WeakEventListener(EventSource source)
  {
    WeakEventManager.AddHandler(source, "Event", OnEvent);
  }
 
  ~WeakEventListener()
  {
    Console.WriteLine("WeakEventListener finalized.");
  }
}

簡單的一行代碼,真簡潔.

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

 

Console.WriteLine("=== .Net 4.5 weak listener (best) ===");
 
EventSource source = new EventSource();
 
WeakEventListener listener = new WeakEventListener(source);
 
source.Raise();
 
Console.WriteLine("Setting listener to null.");
listener = null;
 
TriggerGC();
 
source.Raise();
 
Console.WriteLine("Setting source to null.");
source = null;
 
TriggerGC();

輸出也是肯定正確的:
 

WeakEventListener received event.
Setting listener to null.
Starting GC.
WeakEventListener finalized.
GC finished.
Setting source to null.
Starting GC.
GC finished.

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

結(jié)論

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

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

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

相關(guān)文章

  • C#中Task.Yield的用途深入講解

    C#中Task.Yield的用途深入講解

    這篇文章主要給大家介紹了關(guān)于C#中Task.Yield的用途的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2018-11-11
  • 淺析WPF中Binding的數(shù)據(jù)校驗和類型轉(zhuǎn)換

    淺析WPF中Binding的數(shù)據(jù)校驗和類型轉(zhuǎn)換

    在WPF開發(fā)中,Binding實現(xiàn)了數(shù)據(jù)在Source和Target之間的傳遞和流通,那在WPF開發(fā)中,如何實現(xiàn)數(shù)據(jù)的校驗和類型轉(zhuǎn)換呢,下面就跟隨小編一起學習一下吧
    2024-03-03
  • C# WPF上位機實現(xiàn)和下位機TCP通訊的方法

    C# WPF上位機實現(xiàn)和下位機TCP通訊的方法

    這篇文章主要介紹了C# WPF上位機實現(xiàn)和下位機TCP通訊的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2019-03-03
  • C#用Activex實現(xiàn)Web客戶端讀取RFID功能的代碼

    C#用Activex實現(xiàn)Web客戶端讀取RFID功能的代碼

    由于要在Web項目中采用RFID讀取功能,所以有必要開發(fā)Activex,一般情況下開發(fā)Activex都采用VC,VB等,但對這兩塊不是很熟悉,所以采用C#編寫Activex的方式實現(xiàn)
    2011-05-05
  • 使用C#語言實現(xiàn)的查詢條件界面展開和收起功能

    使用C#語言實現(xiàn)的查詢條件界面展開和收起功能

    這篇文章主要介紹了使用C#語言實現(xiàn)的查詢條件界面展開和收起功能的完美解決方案,需要的朋友可以參考下
    2016-11-11
  • webBrowser執(zhí)行js的方法,并返回值,c#后臺取值的實現(xiàn)

    webBrowser執(zhí)行js的方法,并返回值,c#后臺取值的實現(xiàn)

    下面小編就為大家?guī)硪黄獁ebBrowser執(zhí)行js的方法,并返回值,c#后臺取值的實現(xiàn)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-12-12
  • C# DataTable數(shù)據(jù)遍歷優(yōu)化詳解

    C# DataTable數(shù)據(jù)遍歷優(yōu)化詳解

    這篇文章主要介紹了C# DataTable數(shù)據(jù)遍歷優(yōu)化詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-01-01
  • C#紋理畫刷TextureBrush用法實例

    C#紋理畫刷TextureBrush用法實例

    這篇文章主要介紹了C#紋理畫刷TextureBrush用法,實例分析了紋理畫刷TextureBrush的具體使用技巧,需要的朋友可以參考下
    2015-06-06
  • C#中的委托介紹

    C#中的委托介紹

    這篇文章主要介紹了C#中的委托介紹,本文講解了聲明委托的方式、匿名方法、多播委托等內(nèi)容,需要的朋友可以參考下
    2015-01-01
  • C# 反射(Reflection)的用處分析

    C# 反射(Reflection)的用處分析

    反射(Reflection)是C#里很重要的一個特性,其它語言也有這個特性,比如JAVA。反射這個特性是很實用的,如果使用過struts, hibernate, spring等等這些框架的話,便會知道反射這個特性是多么的強大了。在我接觸過的那些框架中,沒有一個框架是不使用反射的。
    2015-03-03

最新評論