深刻理解:C#中的委托、事件

来源:http://www.cnblogs.com/susufufu/archive/2016/12/11/6160764.html
-Advertisement-
Play Games

C 中的事件還真是有點繞啊,以前用JavaScript的我,理解起來還真是廢了好大勁!剛開始還真有點想不明白為什麼這麼繞,想想和JS的區別,最後終於恍然大悟! C 中事件繞的根本原因: 1. C 的方法,它不是一個類型,它只是其它類型的成員; 2. C 是一個強類型的語言,定義方法時,它的參數必須指 ...


C#中的事件還真是有點繞啊,以前用JavaScript的我,理解起來還真是廢了好大勁!剛開始還真有點想不明白為什麼這麼繞,想想和JS的區別,最後終於恍然大悟!

C#中事件繞的根本原因:

  1. C#的方法,它不是一個類型,它只是其它類型的成員;
  2. C#是一個強類型的語言,定義方法時,它的參數必須指定類型,如public void add(int n){...};

所以,一個方法不能直接作為其它方法的參數,把一個方法名作為參數,無法指定類型啊,會報錯!那我就想啊,既然不能直接傳入,那我傳入整個對象總可以吧,通過傳進來的對象來執行該方法,如下代碼:

using System;
namespace MyEventTest
{
    public class SomeClass
    {
        public void Start(int a) { Console.WriteLine("Go:{0}",a); }
    }
    public class Publisher
    {
        public void StartEvent(int a, SomeClass sc)
        {
            if (sc != null)
            {
                sc.Start(a);  //觸發回調方法
            }
        }
    }
    public class MainClass
    {
        static void Main()
        {
            SomeClass some = new SomeClass();
            Publisher p = new Publisher();
            p.StartEvent(5,some); //Go:5
        }
    }
}

以上方法確實可以,但C#不完全是這樣實現事件的,因為方法的特殊性,C#引入了委托的概念,讓委托對象來代表方法作為其它方法的參數;而事件對象,其實就是一個委托對象。下麵先介紹一下委托:

委托

對應於以上方法:
public void Start(int a) { Console.WriteLine("Go:{0}",a); }
我們可以定義一個委托類型:
public delegate void MyDel(int a);

委托的定義:

  1. 委托MyDel它是一個類型,類型名就是MyDel;定義委托相當於定義一個新類,委托在後臺實現為派生自System.Delegate類。
  2. 定義委托,就是告訴編譯器該委托將表示哪種方法(返回值類型+方法簽名),該方法可以是任意類型的實例方法、靜態方法,只要方法的簽名、返回值類型與委托匹配,那麼該委托的實例就可以引用這些方法。
  3. 使用委托,必須創建該委托的實例,併為它指定要引用的方法,如:MyDel d = some.Start;註意這裡不是some.Start()
  4. 委托對象支持"+","+="來為它添加更多的方法引用,而"-","-="則是刪除引用;
  5. 引用了多個方法的委托就叫多播委托,多播委托派生自基類System.MulticastDelegate類,它是System.Delegate類的子類
  6. 註意:只要委托對象還存在對方法的引用,它就一直占用記憶體哦!我想可以用d = null;來釋放委托對象d;
  7. 可以對委托對象執行調用,如:d(5);它將把調用傳遞給它所引用的方法some.Start(5);,對於多播委托,它將按順序調用它引用的所有方法,但如果其中一個方法拋出異常,且沒在方法內部處理,則將會將異常往外拋出,之後的方法調用將終止。

    使用委托的規則:

  8. 委托是和類一個級別的,可以在能定義類的任何地方定義委托;
  9. 委托不支持繼承;
  10. 可以為委托類型定義任意常見的訪問修飾符;
  11. 委托對象所引用的方法也可以是匿名方法、Lambda 表達式;
  12. 多播委托的返回值類型必須是void,否則就只能得到委托調用的最後一個方法的結果。
  13. 在.NET 4.0中,委托開始支持協變與逆變,這樣一來,定義委托類型時的簽名可以和所要引用的方法的簽名不完全匹配(不同類型之間必須是派生關係)
  14. 委托支持泛型,.NET預定義了兩個泛型版本的委托:
    • Action< T >委托表示引用一個返回值類型為void的方法,根據參數個數存在不同的變體版本;如:Action<in T1, in T2>
    • Func< T >委托表示引用一個帶返回值類型的方法,根據參數個數存在不同的變體版本;如:Func<in T1, out TResult>1個參數T1和返回值類型TResult。

事件

說完了委托的概念,就可以繼續講事件了,因為事件是基於委托的!

事件的概念:

  • 類或對象可以通過事件向其他類或對象通知發生的相關事情。
  • 發送事件的類稱為“發行者”,接收事件的類稱為“訂閱者”。(就是設計模式中的訂閱發佈者模式);
  • 一個事件可以有多個訂閱者。 一個訂閱者可處理來自多個發行者的多個事件。如果一個事件有多個訂閱者,當引發該事件時,會同步調用多個事件處理程式。也可非同步調用。

.NET Framework 類庫中的所有事件均基於 EventHandler 委托,還有泛型版本EventHandler<EventArgs>,這個委托是.NET預定義的,不需要我們定義,可以直接用它來實例化一個事件對象,定義如下:

參數object sender對象是對發佈者的實例的引用,EventArgs e對象主要用來存儲事件數據

public delegate void EventHandler(object sender, EventArgs e); //EventArgs主要用來存儲事件數據

public delegate void EventHandler<TEventArgs>(object sender, EventArgs e);

雖然在自定義的類中的事件可基於任何有效委托類型,但是,通常建議使用.NET預定義事件委托類型讓事件基於 .NET 標準事件模式

下麵是我總結的發佈基於 .NET 標準事件模式的4個步驟:

第1步:在發佈者類中實例化委托事件,並定義一個實例方法,用來調用委托事件(因為委托事件只能通過定義它的類的實例來調用)。

定義發佈者類之前可先定義一個用來存儲事件數據的類(它必須派生於EventArgs基類),如下:

註意:在方法StartEvent()中,聲明瞭一個變數,來保存事件對象的副本,這樣在取得事件對象的副本後,到觸發事件時,這段時間內,這個事件副本就不會受其它線程的影響。如:在此期間,其它線程註銷了回調方法,那麼MyEvent就為null了,這時再觸發事件將引發錯誤。(這就是線程安全的事件,當然還可以通過鎖機制,或者為事件對象始終引用一個空方法)

public class MyEventArgs: EventArgs  //定義存儲事件數據的類
{
    public int Current{get;set;}
}
public class Publisher
{
    public event EventHandler<MyEventArgs> MyEvent; //第1步:實例化委托事件
    public int Sum{get;set;}
    public void StartEvent(int a)  
    {
          var EventCopy = MyEvent; //每次都取一個副本
          MyEventArgs args = new MyEventArgs();
          args.Current = a;
          this.Sum += a;
          if (EventCopy != null)
          {
               EventCopy(this,args);  //調用事件
          }
    }
}

第2步:定義訂閱者類,在該類中定義和委托事件相匹配的方法(事件觸發時,實際要執行的方法)

public class Subscriber
{
     public void Dosomething1(object obj, MyEventArgs e)
     {
            Publisher p = (Publisher)obj;
            Console.WriteLine("Meg: Sum = {0}, Current = {1}", p.sum, e.Current);
     }
     public void Dosomething2(object obj, MyEventArgs e)
     {
     }
}

第3步:在客戶端代碼中,在發佈者類的實例上為委托事件註冊回調方法

public class MainClass
{
    static void Main()
    {
         Publisher p = new Publisher{ Sum = 0 };
         Subscriber sub = new Subscriber();
         p.MyEvent += sub. Dosomething1;  //註冊回調方法
         p.MyEvent += sub. Dosomething2;
         
         p. StartEvent( 5 ); //調用方法,間接觸發事件

         p.MyEvent -= sub. Dosomething1;  //取消註冊
    }
}

要點:事件對象其實就是一個委托對象,把事件當委托來看,就比較容易理解了!不要被Event這個單詞給矇蔽了!

介紹完了!下回將介紹C#中的其它一些較難理解的概念!



您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • C#的事件基於委托,所以先說委托。 一切脫離實際場景的抽象概念新手看上去就像是在扯犢子,不錯,我就是個新手。所以我需要一個實際的場景。 明天剛好考試(商務英語),考試上有兩個角色(class):老師(Teacher)和學生(Student),在考試時間終止的時候,老師會觸發(invoke)一個事件( ...
  • 上篇 《python時間時分秒與秒數的互相轉換》http://www.cnblogs.com/gayhub/p/6154707.html 提到了把時間轉成秒數的方法, 這篇寫寫轉換成秒數後有些什麼用處。 場景1:取N個07:30:00 09:30:33之間的隨機時間。 下麵是我的代碼: 從代碼中可以 ...
  • OpenGL常用函數 glAccum 操作累加緩衝區 glAddSwapHintRectWIN 定義一組被 SwapBuffers拷貝的三角形 glAlphaFunc允許設置alpha檢測功能 glAreTexturesResident 決定特定的紋理對象是否常駐在紋理記憶體中 glArrayElem ...
  • 1、put/checkAndPut 使用checkAndPut,需要先對數據進行驗證,上面的例子中,向row1中的cf:col1寫入數據"E",而驗證的是row1中的cf:col5的值是否為"E",註意這一點,相當於加了條件。 2、使用get讀取數據 參考結果: 3、使用scan獲取數據 4、del ...
  • 工作中發現在oozie中使用sqoop與在shell中直接調度sqoop性能上有很大的差異。為了更深入的探索其中的緣由,開始了oozie的源碼分析之路。今天第一天閱讀源碼,由於沒有編譯成功,不能運行測試用例,直接使用sublime肉眼閱讀,還是挺費勁的。 雖然流程還不是順暢,但是大體上的內容還算是了 ...
  • 所有 ReSherper 的功能都可以使用快捷鍵。大部分功能都有預設快捷鍵,剩下的少數功能可以自定義快捷鍵。 ReSharper 提供了兩種快捷鍵的方式 Visual Studio:這種方式可以減少與 Visual Studio 本身快捷鍵的衝突。 ReSharper 2.0/IntelliJ ID ...
  • 網上有用的資料不多,在一本電子書中摘抄了內容如下 webControls配置節只有一個clientScriptsLocation屬性,此屬性用於指定ASP.NET客戶端腳本的預設存放路徑。這些文件是包含在HTML代碼生成的ASPX頁面時這些需要的客戶端功能,如智能導航和客戶端控制項驗證。 <webCo ...
  • 上周收到本書作者李爭送的一本12月份的新書《微軟開源跨平臺移動開發實踐》。這本書的內容確是超豐富,濃縮了微軟這三年向開源和跨平臺領域的轉變,微軟在開源和跨平臺領域構建出來的一套技術體系。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...