什麼是委託
首先要知道什麼是委託,用最通俗易懂的話來講,你就可以把委託看成是用來執行方法(函數)的一個東西。
如何使用委託
在使用委託的時候,你可以像對待一個類一樣對待它。即先聲明,再執行個體化。只是有點不同,類在執行個體化之後叫對象或執行個體,但委託在執行個體化後仍叫委託。
聲明,如:
1 namespace Vczx.ProCSharp.Exc
2 {
3 delegate double MathsOp( double x );
4 //class defination here
5 }
這就聲明了一個委託,意義:任何一個傳回值為double,且只有一個形參為double的函數,都可以用這個委託來調用。
注意:委託的聲明位置在namespace裡面,類的外面。
其實,委託的聲明也可以在類的裡面,甚至是任何一個可以聲明類的地方。
執行個體化:
首先我們要先有一個滿足委託聲明的方法,假設一個返回一個數的2倍的方法:1class MathsOperations
2{
3 public static double MultiplyBy2( double value )
4 {
5 return value * 2;
6 }
7}
有了這樣一個方法,我們就可以執行個體化一個委託了:
MathsOp operation = new MathsOp( MathsOperations.MultiplyBy2 );
在執行個體化一個委託時,要給它一個參數,這個參數就是委託執行的方法,它可以是靜態方法,也可以是執行個體方法(這一點有別於函數指標,函數指標只能調用靜態方法),如:
MathsOp operation = new MathsOp( new Class1().Method1 );
在執行個體化完一個委託之後,就可以用這個委託來調用方法了:
double result = operation( 1.23 );
例子代碼:
1namespace Vczx.ProCSharp.Exc
2{
3 delegate double MathsOp( double x );
4 class Start
5 {
6 public class MyDelegate
7 {
8 public static double MultiplyBy2( double x )
9 {
10 return x * 2;
11 }
12 }
13 [STAThread]
14 static void Main(string[] args)
15 {
16 MathsOp operation = new MathsOp( MyDelegate.MultiplyBy2 );
17 double x = 1.23;
18 double result = operation( x );
19 Console.WriteLine( "{0} multiply by 2 is {1}", x, result );
20 Console.Read();
21 }
22 }
23}
多路廣播委託
前面使用的委託只包含一個方法調用。調用委託的次數與調用方法的次數相同。如果要調用多個方法,就需要多次顯示調用這個委託。其實委託也可以包含多個方法,這種委託就是多路廣播委託。多路廣播委託派生於System.MulticastDelegate,它的Combine方法允許把多個方法調用連結在一起,我們可以通過+=來向委託添加調用方法,也可以用-=刪除其中的調用方法。如:
1namespace Vczx.ProCSharp.Exc
2{
3 public class MyDelegate
4 {
5 public static void MultiplyBy2( double value )
6 {
7 double result = value * 2;
8 Console.WriteLine( "Multiplying by 2: {0} gives {1}", value, result );
9 }
10
11 public static void Squre( double value )
12 {
13 double result = value * value;
14 Console.WriteLine( "Squaring: {0} gives {1}", value, result );
15 }
16 }
17
18 delegate void MathsOp( double x );
19
20 class Start
21 {
22 [STAThread]
23 static void Main(string[] args)
24 {
25 MathsOp operation = new MathsOp( MyDelegate.MultiplyBy2 );
26 operation += new MathsOp( MyDelegate.Squre );
27 double x = 1.23;
28 operation( x );
29
30 operation -= new MathsOp( MyDelegate.MultiplyBy2 );
31 operation( x );
32
33 Console.Read();
34 }
35 }
36}
輸出:
Multiplying by 2: 1.23 gives 2.46
Squaring: 1.23 gives 1.5129
Squaring: 1.23 gives 1.5129
注意,多路廣播委託聲明時必須返回void,否則傳回值不知道應該送回什麼地方。對此,我做了一個測試:如果不將委託的聲明返回void,則傳回值返回的是最後一個鏈入委託鏈的方法的傳回值,編譯不會出錯。
為什麼要用委託
使用委託使程式員可以將方法引用封裝在委派物件內。然後可以將該委派物件傳遞給可調用所引用方法的代碼,而不必在編譯時間知道將調用哪個方法。與C或C++中的函數指標不同,委託是物件導向,而且是型別安全的。