委托(delegate)



 

委托概述

將方法調用者和目標方法動態關聯起來,委托是一個類,所以它和類是同級的,可以通過委托來掉用方法,不要誤以為委托和方法同級的,方法只是類的成員。委托定義了方法的類型(定義委托和與之對應的方法必須具有相同的參數個數,並且類型相同,返回值類型相同),使得可以將方法當作另一個方法的參數來進行傳遞,這種將方法動態地賦給參數的做法,可以避免在程序中大量使用If-Else(Switch)語句,同時使得程序具有更好的可擴展性。

 

基礎委托(Delegate)

在.Net中聲明委托使用關鍵詞delegate委托具有多種使用方式(以下均為同步委托調用

 1     /// <summary>
 2     /// 普通委托基礎調用方式(同步委托)
 3     /// </summary>
 4     public class Delegates
 5     {
 6         /// <summary>
 7         /// 定義有參無返回值委托
 8         /// </summary>
 9         /// <param name="i"></param>
10         public delegate void NoReturnWithParameters(string o);
11         /// <summary>
12         /// 構造函數實例化
13         /// </summary>
14         public void DemoOne()
15         {
16             NoReturnWithParameters methord = new NoReturnWithParameters(this.Test);
17             methord.Invoke("One-ok");
18         }
19         /// <summary>
20         /// 賦值對象
21         /// </summary>
22         public void DemoTwo()
23         {
24             NoReturnWithParameters methord = this.Test;
25             methord.Invoke("Two-ok");
26         }
27         /// <summary>
28         /// DotNet 2.0 
29         /// </summary>
30         public void DemoThree()
31         {
32             NoReturnWithParameters methord = new NoReturnWithParameters(
33                 delegate (string o)
34                      {
35                          Console.WriteLine("有參無返回值:{0}", o);
36                      }
37             );
38             methord.Invoke("Three-ok");
39         }
40         /// <summary>
41         /// DotNet 3.0 
42         /// </summary>
43         public void DemoFour()
44         {
45             NoReturnWithParameters methord = new NoReturnWithParameters(
46                 (string o) =>
47                     {
48                         Console.WriteLine("有參無返回值:{0}", o);
49                     }
50             );
51             methord.Invoke("Four-ok");
52         }
53         /// <summary>
54         /// 委托約束
55         /// </summary>
56         public void DemoFive()
57         {
58             NoReturnWithParameters methord = new NoReturnWithParameters(
59                 (o) =>
60                 {
61                     Console.WriteLine("有參無返回值:{0}", o);
62                 }
63             );
64             methord.Invoke("Five-ok");
65         }
66         /// <summary>
67         /// 方法只有一行去則掉大括號及分號
68         /// </summary>
69         public void DemoSix()
70         {
71             NoReturnWithParameters methord = new NoReturnWithParameters((o) => Console.WriteLine("有參無返回值:{0}", o));
72             methord.Invoke("Six-ok");
73         }
74         public void DemoSeven()
75         {
76             NoReturnWithParameters methord = (o) => Console.WriteLine("有參無返回值:{0}", o);
77             methord.Invoke("Seven-ok");
78         }
79         /// <summary>
80         /// 定義有參無返回值測試方法
81         /// </summary>
82         /// <param name="o"></param>
83         private void Test(string o)
84         {
85             Console.WriteLine("有參無返回值:{0}", o);
86         }
87         /*
88          * 作者:Jonins
89          * 出處:http://www.cnblogs.com/jonins/
90          */
91     }

 

同步委托&異步委托

同步委托:委托的Invoke方法用來進行同步調用。同步調用也可以叫阻塞調用,它將阻塞當前線程,然后執行調用,調用完畢后再繼續向下進行。

異步委托:異步調用不阻塞線程,而是把調用塞到線程池中,程序主線程或UI線程可以繼續執行。委托的異步調用通過BeginInvokeEndInvoke來實現。

以下為異步委托調用方式:

 1     class Program
 2     {
 3         /// <summary>
 4         /// 定義有參無返回值委托
 5         /// </summary>
 6         /// <param name="i"></param>
 7         public delegate void NoReturnWithParameters(string o);
 8         static void Main(string[] args)
 9         {
10             NoReturnWithParameters methord = new NoReturnWithParameters(Test);
11             Console.WriteLine("主線程執行1");
12             Console.WriteLine("主線程執行2");
13             methord.BeginInvoke("demo-ok", null, null);
14             Console.WriteLine("主線程執行3");
15             Console.WriteLine("主線程執行4");
16             Console.ReadKey();
17         }
18         /// <summary>
19         /// 異步調用委托方法
20         /// </summary>
21         /// <param name="o"></param>
22         static void Test(string o)
23         {
24             Console.WriteLine("有參無返回值:{0}", o);
25         }
26         /*
27          * 作者:Jonins
28          * 出處:http://www.cnblogs.com/jonins/
29          */
30     }

因為調用BeginInvoke為異步委托,不會阻塞主線程,運行結果如下:

 

異步回調(Callback)

異步回調通過設置回調函數,當調用結束時會自動調用回調函數,可以在回調函數里觸發EndInvoke,這樣就釋放掉了線程,可以避免程序一直占用一個線程。

 1     class Program
 2     {
 3         /// <summary>
 4         /// 定義有參有返回值委托
 5         /// </summary>
 6         /// <param name="i"></param>
 7         public delegate string ReturnWithParameters(string o);
 8         static void Main(string[] args)
 9         {
10             ReturnWithParameters methord = new ReturnWithParameters(Test);
11             Console.WriteLine("主線程執行1");
12             Console.WriteLine("主線程執行2");
13             /*
14              BeginInvoke方法參數個數不確定, 最后兩個參數含義固定,如果不使用的話,需要賦值null
15              委托的方法無參數,這種情況下BeginInvoke中只有兩個參數。
16              此外,委托的方法有幾個參數,BeginInvoke中從左開始,對應響應的參數。
17              1.倒數第二個參數:是有一個參數值無返回值的委托,它代表的含義為,該線程執行完畢后的回調。
18              2.倒數第一個參數:向即回調中傳值,用AsyncState來接受。
19              3.其它參數:對應委托方法的參數。
20              */
21             IAsyncResult asyncResult = methord.BeginInvoke("demo-ok", new AsyncCallback(Callback), "AsycState:給回調函數的參數傳遞在此處出傳值");
22             Console.WriteLine("主線程執行3");
23             Console.WriteLine("主線程執行4");
24             Console.ReadKey();
25         }
26         /// <summary>
27         /// 異步調用委托方法
28         /// </summary>
29         /// <param name="o"></param>
30         /// <returns></returns>
31         private static string Test(string o)
32         {
33             return "委托方法執行成功:" + o;
34         }
35         /// <summary>
36         /// 回調函數
37         /// </summary>
38         /// <param name="asyncResult"></param>
39         private static void Callback(IAsyncResult asyncResult)
40         {
41             /*
42              *asyncResult為回調前異步調用方法返回值
43              *AsyncResult 是IAsyncResult接口的一個實現類,引用空間:System.Runtime.Remoting.Messaging
44              *AsyncDelegate 屬性可以強制轉換為定義的委托類型
45              */
46             ReturnWithParameters methord = (ReturnWithParameters)((System.Runtime.Remoting.Messaging.AsyncResult)asyncResult).AsyncDelegate;
47             Console.WriteLine(methord.EndInvoke(asyncResult));
48             Console.WriteLine(asyncResult.AsyncState);
49         }
50         /*
51          * 作者:Jonins
52          * 出處:http://www.cnblogs.com/jonins/
53          */
54     }

執行結果如下:

注意:

1.異步調用只能調用一次EndInvoke,否則會報錯。

2.如果不回調函數中執行EndInvoke,請在異步調用后手動執行EndInvoke方法釋放資源。

 

異步委托線程等待 

1.【Delegate】.EndInvoke(推薦)

1   public delegate void NoReturnWithParameters(string o);
2   NoReturnWithParameters noReturnWithParameters = new NoReturnWithParameters(...);
3        ......
4   noReturnWithParameters.EndInvoke(asyncResult);

2.【IAsyncResult】.AsyncWaitHandle.WaitOne(可以定義等待時間,超過等待時間不繼續等待向下執行)

1  IAsyncResult asyncResult = null;
2  asyncResult.AsyncWaitHandle.WaitOne(2000);//等待2000毫秒,超時不等待

3.【IAsyncResult】.IsCompleted(是IAsyncResult對象的一個屬性,該值指示異步操作是否已完成。不推薦)

1  IAsyncResult asyncResult = xxx.BeginInvoke(...);
2  while (!asyncResult.IsCompleted)
3  {
4      //正在等待中
5  }

 

內置委托(泛化委托)

 .Net Framework 提供兩個支持泛型的內置委托,分別是Action<>Func<>,在System命名空間中定義,結合lambda表達式,可以提高開發效率。

使用方式如下:

 1     class Program
 2     {
 3         static void Main(string[] args)
 4         {
 5             //使用Action聲明委托
 6             Action<string> action = TestAction;
 7             action.Invoke("action-demo-ok");
 8             //使用Func聲明委托
 9             Func<string, string> func = TestFunc;
10             string result = func.Invoke("func-demo-ok");
11             Console.WriteLine(result);
12             Console.ReadKey();
13         }
14         private static void TestAction(string o)
15         {
16             Console.WriteLine("TestAction方法執行成功:{0}", o);
17         }
18         private static string TestFunc(string o)
19         {
20             return "TestFunc方法執行成功:" + o;
21         }
22         /*
23          * 作者:Jonins
24          * 出處:http://www.cnblogs.com/jonins/
25          */
26     }

Action:無返回值的泛型委托,目前.NET Framework提供了17個Action委托,它們從無參數到最多16個參數。

public delegate void Action
Action 無返回值的泛型委托
Action<int,string> 傳入參數int、string,無返回值的委托
Action<int,string,bool>  傳入參數int,string,bool,無返回值的委托
Action<bool,bool,bool,bool>  傳入4個bool型參數,無返回值的委托
Action最少0個參數,最多16個參數,無返回值。

 

 

 

 

 

Func:有返回值的泛型委托,.NET Framework提供了17個Func函數,允許回調方法返回值。

public delegate TResult Func
Func<int>  無參,返回值為int的委托
Func<int,string> 傳入參數int,返回值為string類型的委托
Func<object,string,bool>  傳入參數為object, string 返回值為bool類型的委托
Func<T1,T2,,T3,int> 表示 傳入參數為T1,T2,,T3(類型)返回值為int類型的委托
Func最少0個參數,最多16個參數,根據返回值泛型返回。必須有返回值,不可為void

 

 

 

 

 

本質上ActionFunc都為delegate ,在System命名空間中定義(in和out用來標識變量)

除此之外還有Predicate,它是固定返回值為bool類型的泛型委托。Action和Func足夠使用這里不做介紹。

注意

1.委托定義不要太多,微軟僅在MSCorLib.dll中就有進50個委托類型,而且.NET Framework現在支持泛型,所以我們只需幾個泛型委托(在System命名空間中定義)就能表示需要獲取多達16個參數的方法。

2.如需獲取16個以上參數,就必須定義自己的委托類型。所以建議盡量使用內置委托,而不是在代碼中定義更多的委托類型,這樣可以減少代碼中的類型數量,同時簡化編碼。

3.如需使用ref或out關鍵字以傳引用的方式傳遞參數,就需要定義自己的委托。

 

內置委托(泛化委托)參數協變&逆變

協變(out):假定S是B的子類,如果X(S)允許引用轉換成X(B),那么稱X為協變類。(支持“子類”向“父類”轉換)
逆變(in):假定S是B的子類,如果X(B)允許引用轉換成X(X),那么稱X為協變類。(支持“父類”向“子類”轉換)

正如泛化接口,泛型委托同樣支持協變與逆變

1     public delegate void Action<in T>(T obj);
2    
3     public delegate TResult Func<out TResult>();

Action在System命名空間中定義支持逆變(in)

1         Action<object> x =...;
2         
3         Action<string> y = x;    

Func在System命名空間中定義支持協變(out)

1         Func<string> x =...;
2             
3         Func<object> y = x; 

如果要定義一個泛化委托類型,最好按照如下准則:
1.將只用在返回值的類型參數標注為協變(out)
2.將只用在參數的類型參數標注為逆變(in)

委托的兼容性

了解委托的兼容性,更易於在使用委托時使我們構建的代碼具有多態性

1.類型的兼容性:即使簽名相似,委托類也互不兼容。

1 delegate void D1();
2 delegate void D2();
3 ...
4 D1 d1=Method1;
5 D2 d2=d1;//編譯時錯誤
6 D2 d2=new D2(d1);//這是允許的

如果委托實例執行相同的目標方法,則認為它們是等價的。

1 delegate void D();
2 ...
3 D1 d1=Method1;
4 D2 d2=Method1;
5 Console.WriteLine(d1==d2);//True

如果多播委托按照相同的順序應用相同的方法責任委托它們是等價的。

2.參數的兼容性:當調用一個方法時,可以給方法的參數提供大於其指定類型的變量。這是正常的多態行為。同樣,委托也可以又大於其目標方法參數類型的參數,即逆變。

 1     class Program
 2     {
 3         //委托接受string類型參數
 4         delegate void NoReturnWithParameters(string o);
 5         static void Main(string[] args)
 6         {
 7             NoReturnWithParameters noReturnWithParameters = new NoReturnWithParameters(Test);
 8             noReturnWithParameters("demo-ok");
 9             Console.ReadKey();
10         }
11         //目標方法接受object類型參數
12         static void Test(object o)
13         {
14             Console.WriteLine("返回值:{0}", o);
15         }
16     }

上述代碼將參數string在調用目標方法時隱式向上轉換為Object。

3.返回類型的兼容性:如果調用一個方法,得到的返回值類型可能大於請求的類型,這是正常多態行為。同樣,委托的返回類型可以小於它的目標方法的返回值類型即協變

 1     class Program
 2     {
 3         //委托返回object類型
 4         delegate object NoReturnWithParameters(string o);
 5         static void Main(string[] args)
 6         {
 7             NoReturnWithParameters noReturnWithParameters = new NoReturnWithParameters(Test);
 8             object o = noReturnWithParameters("demo-ok");
 9             Console.WriteLine(o);
10             Console.ReadKey();
11         }
12         //目標方法返回string類型
13         static string Test(string o)
14         {
15             return "返回值:" + o;
16         }
17     }

注意標准事件模式的設計宗旨時再其使用公共基類EventArgs時應用逆變。例如,可以用兩個不同的委托調用同一個方法,一個傳遞MouseEventArgs,另一個傳遞KeyEventArgs。

 

多播委托(+=&-=)

所有的委托的實例都有多播的功能,自定義委托和內置委托都有,可以通過 +=-=給委托增加和刪掉不同的方法,當輸入參數后,每個方法會按順序進行迭代處理,並返回最后一個方法的計算結果。下面是簡單模擬計算器的一段代碼:
 1     class Program
 2     {
 3         public delegate int MulticastInstance(int inputA, int inputB);
 4         static void Main(string[] args)
 5         {
 6             MulticastInstance multicastInstance = Addition;
 7             multicastInstance += new MulticastInstance(Reduce);
 8             multicastInstance += new MulticastInstance(Multiply);
 9             int result = multicastInstance(10, 5);
10             Console.WriteLine("最后執行得到的結果為:{0}", result);
11             Console.ReadKey();
12         }
13         /// <summary>
14         /// 加法
15         /// </summary>
16         /// <param name="inputA"></param>
17         /// <param name="inputB"></param>
18         /// <returns></returns>
19         private static int Addition(int inputA, int inputB)
20         {
21             int result = inputA + inputB;
22             Console.WriteLine("Addition方法執行結果:{0}", result);
23             return result;
24         }
25         /// <summary>
26         /// 減法
27         /// </summary>
28         /// <param name="inputA"></param>
29         /// <param name="inputB"></param>
30         /// <returns></returns>
31         private static int Reduce(int inputA, int inputB)
32         {
33             int result = inputA - inputB;
34             Console.WriteLine("Reduce方法執行結果:{0}", result);
35             return result;
36         }
37         /// <summary>
38         /// 乘法
39         /// </summary>
40         /// <param name="inputA"></param>
41         /// <param name="inputB"></param>
42         /// <returns></returns>
43         private static int Multiply(int inputA, int inputB)
44         {
45             int result = inputA * inputB;
46             Console.WriteLine("Multiply方法執行結果:{0}", result);
47             return result;
48         }
49         /*
50          * 作者:Jonins
51          * 出處:http://www.cnblogs.com/jonins/
52          */
53     }

得到的結果如下:

多播委托本質是:委托是不可變的,因此調用+=或-=的實質是創建一個新的委托實例,並把它賦值給已有變量。所有的委托類型都是從 System.MulticastDelegate派生的,它又繼承自 System.Delegate,c#將委托中使用的+、-、+=、-=都編譯成 System.Delegate的靜態 CombineRemove方法。
 

委托模擬觀察者

能用委托解決的問題,都可以用接口解決。但再下面的情形中,委托可能是比接口更好的選擇:

1.接口內之定義一個方法

2.需要多播能力

3.訂閱者需要多次實現接口

下面代碼是委托的觀察者模式,優點是解耦且符合開放封閉原則

 1 public class MulticastDelegates
 2 {
 3     public delegate int MulticastInstance(int inputA, int inputB);
 4     /// <summary>
 5     /// 模擬觀察者
 6     /// </summary>
 7     public void Demo()
 8     {
 9         Manager manager = new Manager();
10         manager.Attach(new MulticastInstance(Add));
11         manager.Attach(new MulticastInstance(Reduce));
12         manager.Attach(new MulticastInstance(Multiply));
13         manager.Execute(10, 5);
14     }
15     /// <summary>
16     /// Observer模式、又稱呼發布訂閱或監聽模式
17     /// </summary>
18     public class Manager
19     {
20         private MulticastInstance Handler;
21 
22         /// <summary>
23         /// 附加觀察者
24         /// </summary>
25         /// <param name="handler1"></param>
26         public void Attach(MulticastInstance handler1)
27         {
28             Handler += handler1;
29         }
30         /// <summary>
31         /// 分離觀察者
32         /// </summary>
33         /// <param name="handler1"></param>
34         public void Detach(MulticastInstance handler1)
35         {
36             Handler -= handler1;
37         }
38         /// <summary>
39         /// 如果觀察者數量大於0即執行播委托列表中的方法
40         /// </summary>
41         /// <param name="inputA"></param>
42         /// <param name="inputB"></param>
43         public void Execute(int inputA, int inputB)
44         {
45             if (Handler != null)
46                 if (Handler.GetInvocationList().Count() != 0)
47                     Handler(inputA, inputB);
48         }
49     }
50     private int Add(int inputA, int inputB)
51     {
52         int result = inputA + inputB;
53         Console.WriteLine("Add方法執行結果:{0}", result);
54         return result;
55     }
56     private int Reduce(int inputA, int inputB)
57     {
58         int result = inputA - inputB;
59         Console.WriteLine("Reduce方法執行結果:{0}", result);
60         return result;
61     }
62     private int Multiply(int inputA, int inputB)
63     {
64         int result = inputA * inputB;
65         Console.WriteLine("Multiply方法執行結果:{0}", result);
66         return result;
67     }
68 }

 

委托揭秘

委托看似很容易使用,通過 delegate關鍵詞定義,用熟悉的 new構造委托實例,熟悉的方式調用回調函數,但實際上編譯器和CLR在幕后做了大量工作來隱藏其復雜性。
重新審視上面計算器的一段代碼:
1     public delegate int MulticastInstance(int inputA, int inputB);

事實上通過反編譯可看到:

編譯器相當於定義了一個完整的類(繼承自System.MulticastDelegate,定義四個方法:構造函數、Invoke、BeginInvoke和EndInvoke):

 
 1      internal class MulticastInstance : System.MulticastDelegate//繼承System.MulticastDelegate
 2         {
 3             //構造器
 4             public MulticastInstance(object @object, IntPtr method);
 5             //這個方法的原型和源代碼指定的一樣
 6             public virtual int Invoke(int inputA, int inputB);
 7             //實現回調方法和異步回調
 8             public virtual IAsyncResult BeginInvoke(int inputA, int inputB, AsyncCallback callback, object @object);
 9             public virtual int EndInvoke(IAsyncResult result);
10         }
11         /*
12          * 作者:Jonins
13          * 出處:http://www.cnblogs.com/jonins/
14          */

所有委托類型都派生自System.MulticastDelegate類,System.MulticastDelegate派生自System.Delegate,后者又派生自System.Object。歷史原因造成有兩個委托類。
創建的所有委托類型豆漿MulticastDelegate作為基類,個別情況下仍會用到Delegate。Delegate類的兩個靜態方法CombineRemove的簽名都指出要獲取Delegate參數。由於創建的委托類型派生自MulticastDelegate,后者又派生自Delegate,所以委托類型的實例是可以傳遞給這兩個方法的。

MulticastDelegate的三個重要非公共字段

字段 類型 說明
_target System.Object

當委托對象包裝一個靜態方法時,這個字段為null。當委托對象包裝一個實例方法時,這個字段引用的是回調方法要操作的對象。

當委托對象包裝一個實例方法時,這個字段引用的是回調方法要操作的對象。換言之

換言之,這個字段指出要傳給實例方法的隱士參數的值。

_methodPtr System.IntPtr

一個內部的整數值,CLR用它標記要回調的方法。

_invocationList System.Object 該字段通常為null,構造委托鏈時它引用一個委托數組。

Delegate反編譯后可看到靜態方法CombineRemove(委托的+、-、+=、-=編譯后的本質):

 1     [Serializable, ClassInterface(ClassInterfaceType.AutoDual), ComVisible(true), __DynamicallyInvokable]
 2     public abstract class Delegate : ICloneable, ISerializable
 3     {
 4         [ComVisible(true), __DynamicallyInvokable]
 5         public static Delegate Combine(params Delegate[] delegates);
 6         [__DynamicallyInvokable]
 7         public static Delegate Combine(Delegate a, Delegate b);
 8         [SecuritySafeCritical, __DynamicallyInvokable]
 9         public static Delegate Remove(Delegate source, Delegate value);
10     }

 

 結語

同步委托將阻塞當前線程,等待方法執行完畢繼續執行程序,相當於直接調用方法。異步委托是將方法放入線程池中執行並不阻塞主線程。異步委托從根本上說並不是多線程技術(任務Task也一樣),就算異步委托內部將方法塞給線程池去執行也並不能說是開辟新線程執行方法,(線程池一定開辟新線程)這種說法並不嚴謹。委托本質是將調用者和目標方法動態關聯起來,這是或許是我所理解的委托存在的最根本目的吧。

 

參考文獻

CLR via C#(第4版) Jeffrey Richter

C#高級編程(第7版) Christian Nagel

果殼中的C# C#5.0權威指南 Joseph Albahari

......


 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM