自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

一文說通Dotnet的委托

開發(fā) 前端
先簡單說說Delegate的由來。最早在C/C++中,有一個概念叫函數(shù)指針。其實就是一個內(nèi)存指針,指向一個函數(shù)。調(diào)用函數(shù)時,只要調(diào)用函數(shù)指針就可以了,至于函數(shù)本身的實現(xiàn),可以放在其它地方,也可以后實現(xiàn)。

[[378624]]

本文轉(zhuǎn)載自微信公眾號「老王Plus」,作者 老王Plus的老王。轉(zhuǎn)載本文請聯(lián)系老王Plus公眾號。

一、前言

先簡單說說Delegate的由來。最早在C/C++中,有一個概念叫函數(shù)指針。其實就是一個內(nèi)存指針,指向一個函數(shù)。調(diào)用函數(shù)時,只要調(diào)用函數(shù)指針就可以了,至于函數(shù)本身的實現(xiàn),可以放在其它地方,也可以后實現(xiàn)。到了.Net,沒有指針的概念了,但這種方式很實用,所以這個概念也保留了下來,形成了現(xiàn)在的委托Delegate。

另外,在.Net中,也把委托延伸了,與執(zhí)行回調(diào)設(shè)計成了同一種機(jī)制,允許開發(fā)者定義回調(diào)的簽名和類型。

當(dāng)我們聲明一個委托時,編譯器會生成一個從MulticastDelegate派生的類。MulticastDelegate還包含幾個方法,不過因為這些方法是CLR運行時動態(tài)生成的,代碼IL中看不到,也不需要關(guān)心。

委托最大的特性是不需要進(jìn)行強(qiáng)耦合。所以調(diào)用者其實并不知道所調(diào)用的是靜態(tài)方法還是實例方法,也不知道具體調(diào)用的內(nèi)容。舉個常見的例子,UI編程中的按鈕Button類。按鈕類本身并不知道它的OnClick事件是如何處理的,也不需要知道。所以實際中,OnClick事件就是使用委托發(fā)布的。開發(fā)者在開發(fā)過程中實現(xiàn)OnClick事件的處理,并由UI訂閱使用。

這種方式,就是委托對類的解耦。

二、簡單委托

委托有一個非常簡單的規(guī)則,就是:要引用的方法的返回類型或參數(shù)要與委托類型聲明相匹配。

聽著有點繞口,我們拿一個例子來說。

我們有一個方法:

  1. void PrintInfo(string message); 

按照規(guī)則,這個方法對應(yīng)的委托方法可以寫成:

  1. void Delegate_PrintInfo(string message); 

這樣,按照規(guī)則,委托使用時就可以寫成:

  1. Delegate_PrintInfo = PrintInfo; 

這樣,當(dāng)我們調(diào)用Delegate_PrintInfo("Hello WangPlus")的時候,實際執(zhí)行的是PrintInfo("Hello WangPlus")了。

下面,我們來看看委托的聲明。

  1. public delegate int Delegate_Method(int x, int y);   

委托可以封裝任何方法。在上面這個例子里,我們接受兩個參數(shù),并返回一個int值。

在這樣一個聲明中,delegate是一個關(guān)鍵詞,表明我們聲明的是一個委托。而其它部分,跟我們正常的代碼方式?jīng)]有任何區(qū)別。

多舉幾個例子看看:

  1. public delegate void Demo_Func1(string para); 
  2. public delegate ClassA Demo_Func2(ClassB para); 
  3. private delegate StructA Demo_Func3(int para); 

除了delegate,其它內(nèi)容跟正常方法沒有區(qū)別。

聲明有了,如何用呢?看例子:

  1. class Program 
  2.     public delegate int Delegate_Method(int x, int y); 
  3.  
  4.     static void Main(string[] args) 
  5.     { 
  6.         Delegate_Method handler = SumMethod; 
  7.         int result = handler(3, 4); 
  8.     } 
  9.     static int Sum(int x, int y) 
  10.     { 
  11.         return x + y; 
  12.     } 

這是一個簡單的例子。

我們先定義了一個委托,接受兩個參數(shù),并返回int值。我希望這個委托調(diào)用下面的Sum方法,因此Sum方法和委托Delegate_Method的簽名(參數(shù)和返回值)兼容。這兒要注意理解這個兼容的概念,不是完全相同,是兼容。

再寫個稍微復(fù)雜一點的例子:

  1. public delegate void Delegate_Method(int x, int y); 
  2.  
  3. class ExampleClass 
  4.     public void Sum(int x, int y) 
  5.     { 
  6.         Console.WriteLine(x + y); 
  7.     } 
  8.     public void Sub(int x, int y) 
  9.     { 
  10.         Console.WriteLine(x - y); 
  11.     } 
  12. class Program 
  13.     static void Main(string[] args) 
  14.     { 
  15.         ExampleClass example = new ExampleClass(); 
  16.         Delegate_Method delegate_1; 
  17.         Delegate_Method delegate_2; 
  18.  
  19.         delegate_1 = example.Sum
  20.         delegate_2 = example.Sub; 
  21.  
  22.         delegate_1(100, 50); 
  23.         delegate_2(100, 50); 
  24.     } 

如果第一個例子明白了,那這個例子也不難理解。

三、委托鏈

委托鏈的核心的維護(hù)一個可調(diào)用的委托列表。當(dāng)調(diào)用列表時,列表中的所有委托都會被調(diào)用。同時,委托鏈可以使用操作符,用+來組合,用-來刪除。

看例子:

  1. public delegate void Delegate_Method(int x, int y); 
  2.  
  3. class ExampleClass 
  4.     public void Sum(int x, int y) 
  5.     { 
  6.         Console.WriteLine(x + y); 
  7.     } 
  8.     public void Sub(int x, int y) 
  9.     { 
  10.         Console.WriteLine(x - y); 
  11.     } 
  12. class Program 
  13.     static void Main(string[] args) 
  14.     { 
  15.         ExampleClass example = new ExampleClass(); 
  16.         Delegate_Method[] delegate_list = new Delegate_Method[] { example.Sum, example.Sub }; 
  17.         Delegate_Method delegate_chain = delegate_list[0] + delegate_list[1]; 
  18.  
  19.         delegate_chain(100, 50); 
  20.     } 

在這個例子中,定義了一個委托數(shù)組,然后用+操作符組合這些方法。

  1. Delegate_Method delegate_chain = delegate_list[0] + delegate_list[1];   
  2. Delegate_Method delegate_chain1 = delegate_chain - delegate_list[0];   

上面兩行代碼,CLR將解釋為(Sum + Sub) - Sum,并只執(zhí)行Sub方法。這是一個使用-操作符從委托鏈中移除委托的例子。

您還可以遍歷委托鏈:

  1. public delegate void Delegate_Method(int x, int y); 
  2.  
  3. class ExampleClass 
  4.     public void Sum(int x, int y) 
  5.     { 
  6.         Console.WriteLine(x + y); 
  7.     } 
  8.     public void Sub(int x, int y) 
  9.     { 
  10.         Console.WriteLine(x - y); 
  11.     } 
  12. class Program 
  13.     static void Main(string[] args) 
  14.     { 
  15.         ExampleClass example = new ExampleClass(); 
  16.         Delegate_Method[] delegate_list = new Delegate_Method[] { example.Sum, example.Sub }; 
  17.         Delegate_Method delegate_chain = delegate_list[0] + delegate_list[1]; 
  18.  
  19.         Delegate[] delegates = delegate_chain.GetInvocationList(); 
  20.         for (int i = 0; i < delegates.Length; i++) 
  21.         { 
  22.             Delegate_Method _delegate = (Delegate_Method)delegates[i]; 
  23.             _delegate(100, 50); 
  24.         } 
  25.     } 

在這個例子中,使用了GetInvocationList方法獲取委托鏈中的所有委托。這個方法幫助我們引用委托鏈中的每個委托,我們也可以從委托鏈中以任何順序調(diào)用委托。

四、多播委托

委托在被調(diào)用時可以調(diào)用多個方法,這稱之為多播。委托對象的一個非常有用的屬性是,它們可以被分配給一個委托實例,以便使用+/-操作符進(jìn)行多播。組合委托調(diào)用由它組成的多個委托。

多播委托時,只能組合相同類型的委托。操作符可用于從組合委托中增加/刪除委托組件。

此外,多播委托返回類型總是void。

  1. class Program 
  2.     public delegate void Delegate_Method(int x, int y); 
  3.  
  4.     public static void Sum(int i, int j) 
  5.     { 
  6.         Console.WriteLine(i + j); 
  7.     } 
  8.     public static void Sub(int i, int j) 
  9.     { 
  10.         Console.WriteLine(i - j); 
  11.     } 
  12.  
  13.     static void Main(string[] args) 
  14.     { 
  15.         Delegate_Method delegate1, delegate2, delegate3, delegate4; 
  16.  
  17.         delegate1 = Sum
  18.         delegate2 = Sub; 
  19.  
  20.         delegate3 = delegate1 + delegate2; 
  21.         delegate3(100, 50); 
  22.  
  23.         delegate4 = delegate3 - delegate2; 
  24.         delegate4(100, 50); 
  25.     } 

這段代碼里,delegate3 = delegate1 + delegate2;等同于挨個調(diào)用Sum和Sub;delegate4 = delegate3 - delegate2;等同于調(diào)用(Sum + Sub) - Sub,實際最后調(diào)用的是Sum。

五、結(jié)論

委托在Dotnet里,是一個很常用的代碼組成。用好委托,可以很漂亮地實現(xiàn)諸如事件、回調(diào)等操作,所以必須要熟練。

最后再說一下委托的基本內(nèi)容:

  • 委托是面向?qū)ο蟮牟僮?,類型安全,?shù)據(jù)安全;
  • 委托派生自Dotnet的Delegate類,它是一個類;
  • 委托類型是密封(sealed)的,所以不能從委托繼承。

 

責(zé)任編輯:武曉燕 來源: 老王Plus
相關(guān)推薦

2021-09-15 06:55:34

異步LinqC#

2021-04-14 07:47:59

AttributeC#屬性

2022-04-28 10:41:08

SaaS業(yè)務(wù)方式

2019-11-12 15:11:45

秒殺流量高可用

2021-07-31 23:14:26

OpenCL框架語言

2021-12-15 09:32:41

Linux系統(tǒng)負(fù)載

2025-04-22 08:57:27

2018-05-22 10:09:09

數(shù)據(jù)庫MySQL優(yōu)化原理

2019-01-29 09:36:10

MySQLACID特性

2020-05-11 07:57:33

區(qū)塊鏈分布式鏈上

2023-01-26 01:09:31

配置數(shù)據(jù)源參數(shù)

2024-02-22 14:20:44

數(shù)字化轉(zhuǎn)型數(shù)字化

2020-01-22 16:50:32

區(qū)塊鏈技術(shù)智能

2021-02-03 08:12:23

函數(shù)委托Dotnet

2022-10-08 06:38:01

元宇宙NFT加密貨幣

2025-01-14 17:00:00

SpringBoot開發(fā)代碼

2020-03-31 14:40:24

HashMap源碼Java

2024-04-12 12:19:08

語言模型AI

2020-08-27 07:34:50

Zookeeper數(shù)據(jù)結(jié)構(gòu)

2024-02-01 11:57:31

this指針代碼C++
點贊
收藏

51CTO技術(shù)棧公眾號