在學習C#3.0之前還是先來回顧下委托、事件,因為這樣能更加有助於理解 C#3.0裡面的一些新的特性,如Lambada表達式等。
背景
在C語言 中我們可以用函數指針來創建回調函數,但是在C裡面回調函數存在一些安全問 題。因為它只是在內存地址中記錄了下來,並沒有像方法的參數類型、參數個數 、返回值等其他安全信息。而在.Net FrameWork中,回調仍然是可以的,.net framework中提供了更為高級的更為安全的面向對象的delegate來實現。
定義委托
在委托中主要包含了三個重要的信息:
1.調用的方法的 名稱;
2.方法的參數;
3.方法的返回值。
定義一個簡單 的委托:
public delegate int Caculate(int x,int y)
這樣的委托就是簽名了參數為兩個int類型返回值為int類型的參 數,Caculate類型的對象可以在運行時動態地調用其指向的方法。要注意的 是.net委托既可以指向動態的方法也可以指向靜態的方法。
C#編譯器在 處理delegate的時候,它會先自動產生一個繼承於System.MulticastDelegate的 類。正是這樣的類根System.Delegate為委托提供了必要的基礎信息,以便來維 護需要調用的方法列表。我們可以通過IL查看器看到:
生成的Caculate類中定義了三個方法:BeginInvoke、EndInvoke、Invoke。其 中Invoke是核心的方法,它用來以同步的方式調用委托列表中的每個方法。我們 可以看一看編譯器是如何定義這幾個方法的。Invoke方法中的參數跟返回值完全 跟Caculate委托的定義一樣的,而BeginInvoke中多了兩個參數一個是 AsyncCallback類型的一個是object類型的,EndInvoke方法返回Int類型。
通過MulticastDeletate跟Delegate基類獲取更多信息
我通過上 面的IL代碼也可以看到編譯器生成委托時的類是繼承於MulticastDelegate的, 而MulticastDelegate繼承於Delegate類,所以可以通過這兩個類來獲取委托更 多的輔助信息。在這裡只列出一些常用的屬性和方法,可以在msdn上獲取更多地 這兩個類的內容(MulticastDelegate成員)。
1.Methos屬性:返回 System.Reflection.MethodInfo類型,描述委托所表示的方法信息。
2.Target屬性:返回委托方法所在的對象,如果是靜態方法即返回null。
3.GetInvocationList方法:返回一個Delegate類型的數組,其中數組的 每個元素表示一個可以調用的方法。
4.Combine方法: 靜態方法用來給委 托添加一個方法。
5.Remove方法:靜態方法給委托移除某個方法。
看下簡單的Caculate的實現,CaculateClient類:
public class CaculateClient
{
public int Add(int x, int y)
{
return x + y;
}
public int Subtract(int x, int y)
{
return x - y;
}
}
另外定義了一個輔助方法:
public static void DisplayDelegateInfo(Delegate del)
{
foreach (Delegate d in del.GetInvocationList())
{
Console.WriteLine("Method Name:{0}", d.Method.Name);
Console.WriteLine("Target is:{0}", d.Target);
}
}
調 用:
static void Main(string[] args)
{
CaculateClient caClient =new CaculateClient();
Simple.Caculate ca = new Simple.Caculate(caClient.Add);
Console.WriteLine("1+1={0}",ca(1,1));
ca += new Simple.Caculate(caClient.Subtract);
Console.WriteLine("Result:{0}", ca(1, 1));
Simple.DisplayDelegateInfo(ca);
Console.Read();
}
我們可以看到下面的結果:
小結
現在我們基本上知道Delegate的原理,已經基本的實現,但是我 們並沒有實現一些高級的話題,畢竟Caculate還只是一個玩具,呵呵。下一篇中 將會涉及到多播、復雜點的示例以及事件
出處: http://henllyee.cnblogs.com/