C# 基礎知識系列- 11 委托和事件

来源:https://www.cnblogs.com/c7jie/archive/2020/04/20/12740794.html
-Advertisement-
Play Games

0. 前言 事件和委托是C 中的高級特性,也是C 中很有意思的一部分。出現事件的地方,必然有委托出現;而委托則不一定會有事件出現。那為什麼會出現這樣的關係呢?這就需要從事件和委托的定義出發,瞭解其中的內在。 1. 委托 說起委托,就不得不回憶一下之前在Linq篇中介紹的匿名方法,其中提到了Func和 ...


0. 前言

事件和委托是C#中的高級特性,也是C#中很有意思的一部分。出現事件的地方,必然有委托出現;而委托則不一定會有事件出現。那為什麼會出現這樣的關係呢?這就需要從事件和委托的定義出發,瞭解其中的內在。

1. 委托

說起委托,就不得不回憶一下之前在Linq篇中介紹的匿名方法,其中提到了Func和Action這兩個類型。這兩個類型就是委托。

委托在C#中定義為一種面向對象形式的方法定址方案。簡單來講,就是定義一個類型,然後表示這個類型代表某一種方法。而委托對象,就是方法參數化。委托可以實現將方法當做一個參數傳遞給另一個方法,也可以認為是反射中的MethodInfo的一種特例(實際上並沒有太多關係)。

委托不關心方法叫什麼,也不關心方法從哪來(歸屬於哪個類或者哪個對象),只關心方法需要哪些參數,返回什麼類型。

說到這裡,我們來看一下如何定義一個委托吧,委托的定義形式如下:

delegate <返回類型>  委托名(參數列表);//參數列表代表任意個參數

由之前的定義形式,我們可以知道委托也是一種類型,所以它的定義也符合類型的定義規範。現在我們定義一個沒有返回值也沒有參數類型的委托作為我們創建的第一個委托:

public delegate void FirstDel();// 類型名稱是 FirstDel

簡單的使用一下:

FirstDel del ;
del();// 會直接報錯

上述代碼如果運行的話,會很直接的報錯,因為你沒有告訴編譯器變數del 應該是什麼,也就是沒有為del賦值,同時委托可以賦值為null,所以在使用的時候需要註意不能為null,否者也是無法運行的。

這裡應用匿名方法的話,可以按照下麵的代碼對del進行賦值:

del = ()=>
{
 	//省略方法   
}

那麼我們熱身結束,開始正式創建一個有意義的委托:

public delegate decimal CalculateArea(decimal height, decimal weight);

上述委托聲明瞭一個計算面積的規範,使用長寬進行面積計算,那麼我們來為它賦值:

CalculateArea squrare = (height, weight) => height * height;// 正方形
CalculateArea rectangle = (height, weight) => height * weight;// 矩形
CalculateArea triangle = (height, weight) => height * weight / 2; //三角形

我們依次創建了三個計算面積的方法,分別是正方形、矩形、三角形,分別調用它們將會得到對應的計算結果:

var squrareArea = squrare(10, 10);// 100
var rectangleArea = rectangle(19, 10);//190
var triangleArea = triangle(10, 5);//25

特別的,C#中委托支持多路廣播,所以也可以使用+-進行註冊和刪除。多路廣播是指在事件和委托中有多個監聽器或響應方法,當事件觸發或者委托調用的時候,註冊的方法組將會都調用。當使用這種方式對委托進行賦值的時候,委托將自動轉為方法組,簡單理解就是 委托對象內部創建了一個列表,然後把賦值給它的方法都存進去了。

所以就會產生如下操作:

CalculateArea calculate = squrare;// calculate必須先賦值一個方法
calculate += rectangle;// 增加 矩形的面積計算方法
calculate += triangle; // 增加三角形的面積計算方法
calculate -= triangle; // 減去三角形的面積計算方法

到這裡會產生一個疑問,calculate運行結果是什麼,會返回一個數組或者其他類型嗎?顯然不會,因為calculate定義的返回類型就是一個decimal,所以不會返回其他的值。

嗯,這就產生了另一個疑問,返回的是哪一個方法的計算結果呢,其他方法的計算結果呢?這裡告訴大家一個結果,只會返回最後一次註冊的方法的執行結果,其他的方法執行了,但是方法的執行結果無法用變數接到。

所以這裡有一個很重要的實踐,如果有需要把委托當做一個方法列表進行使用的時候,最好聲明為void或者拋棄返回值的具體內容。

2. 事件

事件,event。在C#中,事件就像是一種機制,在程式運行到一定階段的時候或者遇到某些狀況的時候,就會觸發一個事件。然後如果有其他代碼訂閱了這個事件,就會自動執行訂閱的代碼。描述起來很抽象,簡單來講就是在類聲明一個委托,並標記這個委托是一個事件,在另一個方法中執行這個事件。其中,觸發這個事件的類稱為發佈者,接受或者註冊了處理方法的類稱為訂閱者。

如何創建或聲明一個事件?聲明一個事件有兩種方式,一種是直接使用EventHandler ,另一種是自己先定義一個委托,然後用這個委托定義事件。

1. 使用EventHandler

public class EventDemo
{
    public event EventHandler HandlerEvent;
}

2. 使用自定義委托

public class EventDemo
{
    public delegate void EventDelegate(object sender, EventArgs e);
    public event EventDelegate DelegateEvent;
}

一般事件的定義約定俗稱是一個void方法,第一個參數是sender表示事件的發佈者,預設是object類型,第二個參數是EventArgs類型的事件變數,表示觸發事件時需要訂閱者註意的內容,一般用來傳一些參數。

其中 EventHandler有一個泛型版本,其聲明如下:

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

其第二個參數並沒有對TEventArgs進行限制,所以我們可以用任何類型當做事件變數。

我們再來看看,EventArgs里有什麼,什麼都沒有,只有一個預設構造方法和幾個繼承自Object的方法。所以在開發中,我們會自己定義一個事件變數類型,為了保持一致會繼承EventArgs。

C#建議事件的定義以On開頭,表示在什麼時觸發,示例代碼並不符合這個規範。

3. 使用一下事件和委托

創建一個帶事件的類:

public class EventDemo
{
    public delegate void EventDelegate(object sender, EventArgs e);

    public event EventDelegate DelegateEvent;


    public void Trigger()
    {
        if (DelegateEvent != null)// 觸發事件,按需判斷事件的訂閱者列表是否為空
        {
            DelegateEvent(this, new EventArgs());
        }
    }
}

使用一下:

EventDemo demo = new EventDemo(); 
demo.DelegateEvent += (sender, eventArgs) =>
{
    //省略訂閱者的方法內容
}
demo.Trigger();//觸發事件          

當發佈者嘗試觸發事件的時候,訂閱者將會接收到消息,然後註冊訂閱者方法就會被調用。發佈者向訂閱者傳遞一對sender和eventArgs,訂閱者按照自己的邏輯進行處理。

這裡很明顯可以看出,事件的處理程式註冊方法用的+=,所以與之對應的也有一個-=表示取消訂閱。

到這裡,委托和事件的基本概念就已經介紹完畢了,當然還是那句話,更多的內容在實踐中。C#的事件機制讓程式員有更多的自由去自定義事件,而不是被局限在某些框架內。所以大家可以多試試C#的事件,也許能發現更多的我不知道的內容呢。

更多內容煩請關註我的博客

file


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

-Advertisement-
Play Games
更多相關文章
  • Java IO流學習總結一:輸入輸出流 轉載請標明出處:http://blog.csdn.net/zhaoyanjun6/article/details/54292148本文出自【趙彥軍的博客】 感謝博主,感謝分享 Java流類圖結構: 流的概念和作用: 流是一組有順序的,有起點和終點的位元組集合,是 ...
  • 單例模式(Singleton Pattern)是一種常用的軟體設計模式,該模式的主要目的是確保某一個類只有一個實例存在。當你希望在整個系統中,某個類只能出現一個實例時,單例對象就能派上用場。 比如,某個伺服器程式的配置信息存放在一個文件中,客戶端通過一個 AppConfig 的類來讀取配置文件的信息 ...
  • 1、雙下方法 定義:雙下方法是特殊方法,它是解釋器提供的 由雙下劃線加方法名加雙下劃線 方法名的具有特殊意義的方法,雙下方法主要是python源碼程式員使用的,我們在開發中儘量不要使用雙下方法,但是深入研究雙下方法,更有益於我們閱讀源碼。 (1)調用:不同的雙下方法有不同的觸發方式, __ len_ ...
  • 在面向對象的中,類與類之間存在三種關係:依賴關係、組合關係、繼承關係。 1、依賴關係: 將一個類的類名或對象當做參數傳遞給另一個函數被使用的關係就是依賴關係 2、組合關係: 將一個類的對象封裝到另一個類的對象的屬性中,就叫組合 3、繼承關係 (1)什麼是面向對象的繼承 繼承(英語:inheritan ...
  • 版權聲明:本文為CSDN博主「Crystal7003」的原創文章,遵循 CC 4.0 BY-SA 版權協議,轉載請附上原文出處鏈接及本聲明。原文鏈接:https://blog.csdn.net/crystal7003/article/details/80043283 感謝博主分享 集合包含Colle ...
  • 以前在 Servlet 中獲取某個指定的 Cookie 的值使用 來獲得所有 Cookie 的值,然後再遍歷。 在 SpringMVC 中可以直接使用 註解來獲得指定的 Cookie 的值。 @CookieValue 中的參數有三個,其中一個 value 用來指定 Cookie 中的參數名,其他參數 ...
  • 怎樣實現WPF Prism Module的國際化和本地化? "English" | 簡體中文 上一篇有簡單介紹主工程的國際化,使用的資源字典(XAML)實現的。 這幾天我添加了幾個Prism模塊(Module),發現子模塊使用資源字典的方式實現國際化和本地化不好做,沒有找到比較好的參考文章,所以換了 ...
  • 最近遇到一個需求,利用樹莓派去採集一個串口設備的所有數據,設備會主動上報數據,但是呢這個設備是一個集合設備,會上報的報文頭都不一樣,比如燈亮度或者開關會上報21 12 ·········,風速會上報71 23 ······。等等10多種數據格式。 剛好net core 支持跨平臺串口通信。所以記錄如 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...