成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

深入理解C#委托的實質(zhì)

開發(fā) 后端
本文重點剖析C#委托的實質(zhì)。委托在本質(zhì)上仍然是一個類,正如很多資料上所說的,委托是一種類型安全的函數(shù)回調(diào)機制。

本文是博客園麒麟.NET的《把委托說透》系列的第二篇,重點剖析C#委托的實質(zhì)。

委托在本質(zhì)上仍然是一個類,我們用delegate關(guān)鍵字聲明的所有委托都繼承自System.MulticastDelegate。后者又是繼承自System.Delegate類,System.Delegate類則繼承自System.Object。委托既然是一個類,那么它就可以被定義在任何地方,即可以定義在類的內(nèi)部,也可以定義在類的外部。

正如很多資料上所說的,委托是一種類型安全的函數(shù)回調(diào)機制, 它不僅能夠調(diào)用實例方法,也能調(diào)用靜態(tài)方法,并且具備按順序執(zhí)行多個方法的能力。

C#委托揭秘

把委托說透(1)中可以看到,委托的使用其實是很簡單的。盡管如此,其內(nèi)部實現(xiàn)仍然相當復(fù)雜。.NET強大的編譯器和CLR掩蓋了這種復(fù)雜性。

為了解釋方便,我們把(1)中的委托代碼復(fù)制在下面,并做一處小小的改動,將LogToTextFile設(shè)置為實例方法。

  1. namespace DelegateSample  
  2. {  
  3.  
  4.     public delegate void Log(string message);  
  5.  
  6.     class UserService  
  7.     {  
  8.         public Log LogDelegate { getset; }  
  9.  
  10.         public UserService() { }  
  11.  
  12.         public void Register(User user)  
  13.         {  
  14.             if (user.Name == "Kirin")  
  15.             {  
  16.                 LogDelegate("注冊失敗,已經(jīng)包含名為" + user.Name + "的用戶");  
  17.             }  
  18.             else 
  19.             {  
  20.                 LogDelegate("注冊成功!");  
  21.             }  
  22.         }  
  23.     }  
  24.  
  25.     class Program  
  26.     {  
  27.         static void Main(string[] args)  
  28.         {  
  29.             User user = new User { Name = "Kirin", Password = "123" };  
  30.             UserService service = new UserService();  
  31.             service.LogDelegate = LogToConsole;  
  32.             Program p = new Program();  
  33.             service.LogDelegate += p.LogToTextFile;  
  34.             service.Register(user);  
  35.  
  36.             Console.ReadLine();  
  37.         }  
  38.  
  39.         static void LogToConsole(string message)  
  40.         {  
  41.             Console.WriteLine(message);  
  42.         }  
  43.  
  44.         void LogToTextFile(string message)  
  45.         {  
  46.             using (StreamWriter sw = File.AppendText("log.txt"))  
  47.             {  
  48.                 sw.WriteLine(message);  
  49.                 sw.Flush();  
  50.                 sw.Close();  
  51.             }  
  52.         }  
  53.     }  
  54. }  
  55.  

打開Reflector反編譯Log委托,可以看到Log類被編譯為如下形式:

反編譯Log委托

在上圖中可以得出如下結(jié)論:

委托是一個類

可以很清晰的看出Log—>MulticastDelegate—>Delegate這種繼承機制。

盡管委托繼承自System.MulticastDelegate類,但我們并不能顯示地聲明一個繼承自System.MulticastDelegate類的委托。委托必須使用delegate關(guān)鍵字聲明,編譯器會自動為我們生成繼承代碼。

由于委托繼承自System.MulticastDelegate類,自然也繼承MulticastDelegate類的字段、屬性和方法。這些成員中,最重要的當屬三個非公共字段,如下表所示:

字段名稱 字段類型 描述
_target System.Object 該字段指明委托所調(diào)用的方法所在的實例類型。如果委托調(diào)用的為靜態(tài)方法,該字段為null;如果為實例方法則為該方法所在的對象。
_methodPtr System.IntPtr 標識回調(diào)方法的指針。
_invocationList System.Object 在構(gòu)建委托鏈時指向一個委托數(shù)組,在委托剛剛構(gòu)建時通常為null。

由上表可以看出,每個委托對象實際上是對方法及其調(diào)用時操作的對象的封裝。MulticastDelegate類還定義了兩個只讀公有實例屬性:Target和Method,分別對應(yīng)_target和_methodPtr。Target屬性返回一個方法回調(diào)時操作的對象引用。如果是靜態(tài)方法則返回null。Method屬性返回一個標識回調(diào)方法的System.Reflection.MethodInfo對象。

編譯器自動為委托創(chuàng)建了BeginInvoke、EndInvoke和Invoke三個方法

當我們在像調(diào)用普通的方法一樣調(diào)用委托時,如

  1. LogDelegate("注冊失敗,已經(jīng)包含名為" + user.Name + "的用戶"); 

這時實際上調(diào)用的是編譯器自動生成的Invoke方法

  1. LogDelegate.Invoke("注冊失敗,已經(jīng)包含名為" + user.Name + "的用戶"); 

使用IL DASM查看UserService的IL代碼,可以驗證以上結(jié)論,如下圖所示:

使用IL DASM查看UserService的IL代碼

在使用委托時,我們也可以顯示調(diào)用Invoke方法(CLR 2.0)。

顯示調(diào)用Invoke方法

Invoke方法的參數(shù)和返回值與委托是一致的。在調(diào)用Invoke方法時,會使用_target和_methodPtr字段。

BeginInvoke和EndInvoke方法用來實現(xiàn)異步調(diào)用,本文在此不進行討論。

委托鏈

委托鏈是一個委托的集合,它允許我們調(diào)用這個集合中的委托所代表的所有方法(對于有返回值的方法,委托鏈的返回值為鏈表中最后一個方法的返回值,本文后面會有詳細介紹)。在Delegate類中定義了3個靜態(tài)方法來幫助我們操作委托鏈。

  1. public static Delegate Combine(params Delegate[] delegates);  
  2. public static Delegate Combine(Delegate a, Delegate b);  
  3. public static Delegate Remove(Delegate source, Delegate value);  

要理解委托鏈,我們首先基于前面的例子,重新聲明兩個委托:logDel1和logDel2。

  1. Log logDel1 = LogToConsole;  
  2. Program p = new Program();  
  3. Log logDel2 = p.LogToTextFile;  

這兩個委托的_target、_methodPtr和_invocationList值分別如下圖所示:

兩個委托的_target、_methodPtr和_invocationList值

構(gòu)造委托鏈

然后,我們使用Combin方法來構(gòu)造一個委托鏈:

  1. Log logChain = null;  
  2. logChain = (Log)Delegate.Combine(logChain, logDel1);  

由于logChain初始為null,在使用Combin方法構(gòu)造委托鏈時,將返回另外一個參數(shù)logDel1,再將logDel1的引用賦給logChain。這時logChain將指向logDel1所指向的對象。

logChain將指向logDel1所指向的對象

接下來我們將logDel2也添加到logChain中來:

  1. logChain = (Log)Delegate.Combine(logChain, logDel2); 

此時,由于logChain已經(jīng)不再是null,將重新構(gòu)建一個新的委托對象。該委托對象的_target和_methodPtr字段與logDel2(第二個參數(shù))相同,_invocationList字段將指向一個委托數(shù)組。該委托數(shù)組中包含兩個元素,第一個元素(索引為0)指向封裝了LogToConsole方法的委托(即logDel1指向的委托);第二個元素(索引為1)指向封裝了LogToTextFile方法的委托(即logDel2指向的委托)。最后,將這個新創(chuàng)建的委托對象的引用賦給logChain。

將這個新創(chuàng)建的委托對象的引用賦給logChain

若再將一個新的委托l(wèi)ogDel3添加到委托鏈中,則仍然會構(gòu)建一個新的委托對象,并將logDel3的引用添加到該委托對象_invocationList的末尾(此時鏈表共有3個元素)。然后,再將該委托對象的引用賦給logChain。而logChain之前指向的委托對象則等待垃圾回收。

至此,委托鏈構(gòu)造完畢,我們來看看如何執(zhí)行委托鏈表中的委托。由于logChain仍然指向一個委托對象,因此執(zhí)行委托鏈表的語法與執(zhí)行委托是一樣的:

  1. logChain("執(zhí)行委托鏈"); 

與普通的委托(如logDel1)所不同的是,logChain的_invocationList字段不為null。這時將首先遍歷執(zhí)行_invocationList中的所有委托。所執(zhí)行的方法的順序與添加的順序一致,依次為LogToConsole、LogToTextFile。

委托Log的Invoke方法的實現(xiàn)用偽代碼表示如下:

  1. public void Invoke(string message)  
  2. {   
  3.     Delegate[] delegateSet = _InvocationList as Delegate[];  
  4.     if (delegateSet != null)   
  5.     {  
  6.         // 如果委托數(shù)組不為空,則依次執(zhí)行該委托數(shù)組中的委托  
  7.         foreach (Feedback d in delegateSet)  
  8.             d(value);  
  9.     }   
  10.     else   
  11.     {  
  12.         // 如果委托數(shù)組為空,則該委托不代表一個委托鏈  
  13.         // 按照正常方式執(zhí)行該委托  
  14.         _methodPtr.Invoke(_target, value);  
  15.     }  
  16. }  

 包含返回值的委托的Invoke實現(xiàn)如下,假設(shè)返回值為string:

  1. public string Invoke(string message)  
  2. {  
  3.     string result = null;  
  4.     Delegate[] delegateSet = _InvocationList as Delegate[];  
  5.     if (delegateSet != null)  
  6.     {  
  7.         // 如果委托數(shù)組不為空,則依次執(zhí)行該委托數(shù)組中的委托  
  8.         foreach (Feedback d in delegateSet)  
  9.             result = d(value);  
  10.     }  
  11.     else 
  12.     {  
  13.         // 如果委托數(shù)組為空,則該委托不代表一個委托鏈  
  14.         // 按照正常方式執(zhí)行該委托  
  15.         result = _methodPtr.Invoke(_target, value);  
  16.     }  
  17.     return result;  
  18. }  

可以看到在委托鏈中,返回值為鏈表中最后一個委托的返回值。

那么如果對兩個委托鏈調(diào)用Combine方法呢?

  1. Log logChain = null;  
  2. Log logChain1 = null;  
  3. Log logChain2 = null;  
  4. logChain1 = (Log)Delegate.Combine(logChain1, logDel1);  
  5. logChain1 = (Log)Delegate.Combine(logChain1, logDel2);  
  6. logChain2 = (Log)Delegate.Combine(logChain2, logDel3;  
  7. logChain2 = (Log)Delegate.Combine(logChain2, logDel4;  
  8. logChain = (Log)Delegate.Combine(logChain1, logChain2);  

最終的結(jié)果是,logChain的_target和_methodPtr均與logDel4相同(確切地說,兩個委托對象的_methodPtr字段并不相同,但Method屬性是相同的),而_invocationList中委托的順序依次為logDel1、logDel2、logDel3、logDel4。

綜上所述,可以對Delegate.Combine(Delegate A, Delegate B)方法做如下總結(jié):

1. 如果A和B均為null,則返回null。

2. 如果A或B一個為null而另一個不為null,則返回不為null的委托。

3. 如果A和B均不為null,返回一個新的委托,該委托

    (1)_target字段與B的_target字段的值相同

    (2)Method屬性與B的Method屬性的值相同

    (3)_invocationList字段為一個委托數(shù)組,該數(shù)組中委托的順序為:A中_invacationList所指向的委托數(shù)組 + B中_invacationList所指向的委托數(shù)組。

移除委托鏈

Combine方法用來向委托鏈中添加一個委托,而Remove方法用來從委托鏈中移除一個委托。

logChain = (Log)Delegate.Remove(logChain, new Log(LogToConsole));

當調(diào)用Remove時,會遍歷(倒序)第一個參數(shù)(logChain)中的中的委托列表(_invocationList字段), 找到與第二個參數(shù)(new Log(LogToConsole))的_target和_methodPtr字段相匹配的委托,并將其從委托列表中移除。返回值需分以下幾種情況,為了描述方便,我們將logChain記為A,將new Log(LogToConsole)記為B。

1. 如果A為null,返回null。

2. 如果B為null,返回A。

3. 如果A的_invocationList為null,即不包含委托鏈,那么如果A本身與B匹配,則返回null,否則返回A。

4. 如果A的_invocationList中不包含與B匹配的委托,則返回A。

5. 如果A的_invocationList中包含與B匹配的委托,則從鏈表中移除B,然后

    (1)如果A的鏈表中只剩下一個委托,則返回該委托。

    (2)如果A的鏈表中還剩下多個委托,將重新構(gòu)建一個新的委托R(R的_invocationList字段為A的_invocationList移除了B之后的鏈表),并返回R。

注意,Remove方法只移除源委托的_invocationList列表中第一個匹配的委托,要想移除所有匹配的委托,可以使用RemoveAll方法。

有了委托鏈,在(1)中提出的第二個疑問就迎刃而解了。當用戶希望使用多種日志記錄方式的時候,使用委托鏈可以輕松地添加和刪除某種日志記錄方式,從而避免了人為地維護一個列表。

總結(jié)

本文首先介紹了C#委托的實質(zhì),委托是一個類,它繼承自System.MulticastDelegate,而MulticastDelegate又繼承自System.Delegate。然后重點剖析了委托鏈,討論了如何創(chuàng)建和移除委托鏈。

【編輯推薦】

  1. C#委托實例簡單分析
  2. 一個.NET委托的故事:彼得,老板和宇宙
  3. 解惑答疑:C#委托和事件
  4. 各版本.NET委托的寫法回顧
  5. 換一個角度看.NET中的理解委托和事件
責(zé)任編輯:yangsai 來源: 博客園
相關(guān)推薦

2024-06-25 08:43:25

C#編程模型

2025-01-09 12:01:53

2024-05-17 12:56:09

C#編程線程

2024-05-11 07:13:33

C#Task編程

2009-01-20 09:54:13

C# 3.0C#改進

2024-10-11 11:54:14

C#編寫異步

2024-06-25 08:33:48

2024-04-10 12:14:36

C++指針算術(shù)運算

2025-01-15 09:34:02

C#屬性方法

2009-08-20 18:11:08

C#異步委托

2016-12-08 15:36:59

HashMap數(shù)據(jù)結(jié)構(gòu)hash函數(shù)

2010-06-01 15:25:27

JavaCLASSPATH

2020-07-21 08:26:08

SpringSecurity過濾器

2024-07-15 08:21:26

TCPC#連接

2022-05-06 16:18:00

Block和 C++OC 類lambda

2012-11-22 10:11:16

LispLisp教程

2023-10-19 11:12:15

Netty代碼

2021-02-17 11:25:33

前端JavaScriptthis

2009-09-25 09:14:35

Hibernate日志

2013-09-22 14:57:19

AtWood
點贊
收藏

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

主站蜘蛛池模板: 国产视频福利一区 | 国产精品18久久久 | 成人精品毛片 | 一级做受毛片免费大片 | 综合国产| 亚洲网站在线 | 九九精品在线 | 亚洲成人av | 欧美极品少妇xxxxⅹ免费视频 | 久久精品99久久 | 久久久美女 | 成人av电影天堂 | 国产日韩欧美一区二区 | 亚洲第一天堂 | 天堂中文av | 久久国产综合 | 天堂免费| 日韩在线资源 | 色天堂影院 | 欧美黄色一区 | 久久久蜜桃 | 欧美激情久久久 | 麻豆久久 | 国产成在线观看免费视频 | 欧美一区二区三区在线观看 | 国产精品伦理一区二区三区 | 国产成人网 | 成人在线免费视频 | 国产婷婷色综合av蜜臀av | 色视频免费 | 免费观看色 | 精品一区二区三区在线视频 | 亚洲www啪成人一区二区 | 亚洲大片在线观看 | 一区二区三区四区在线视频 | 亚洲 中文 欧美 日韩 在线观看 | 天堂在线网| 颜色网站在线观看 | 精品久久久久久久 | 国产精品一级 | 91av视频在线免费观看 |