1. 程式人生 > >C#委托使用詳解(Delegates)

C#委托使用詳解(Delegates)

新的 返回值 一個 個人 執行 第一個 urn 語法 垃圾回收

摘要

委托是C#編程一個非常重要的概念,也是一個難點。本文將系統詳細講解委托。

1. 委托是什麽?

其實,我一直思考如何講解委托,才能把委托說得更透徹。說實話,每個人都委托都有不同的見解,因為看問題的角度不同。個人認為,可以從以下2點來理解:

(1) 從數據結構來講,委托是和類一樣是一種用戶自定義類型

(2) 從設計模式來講,委托(類)提供了方法(對象)的抽象。

既然委托是一種類型,那麽它存儲的是什麽數據?

我們知道,委托是方法的抽象,它存儲的就是一系列具有相同簽名和返回回類型的方法的地址。調用委托的時候,委托包含的所有方法將被執行。

2. 委托類型的定義

委托是類型,就好像類是類型一樣。與類一樣,委托類型必須在被用來創建變量以及類型對象之前聲明。

delegate void MyDel(int x);

委托類型聲明:

(1) 以deleagate關鍵字開頭。

(2)返回類型+委托類型名+參數列表。

3. 聲明委托變量

MyDel del1,del2;

4. 初始化委托變量

(1) 使用new運算符

new運算符的操作數的組成如下:

  • 委托類型名
  • 一組圓括號,其中包含作為調用列表中的第一個成員的方法的名字。方法可以是實例方法或靜態方法。
del1 = new MyDel( myInstObj.MyM1 );
del2 = new MyDel( SClass.OtherM2 );

(2)使用快捷語法

快鍵語法,它僅由方法說明符構成。之所以能這樣,是因為在方法名稱和其相應的委托類型之間有隱式轉換。

del1 = myInstObj.MyM1;
del2 = SClass.OtherM2;

5. 賦值委托

由於委托是引用類型,我們可以通過給它賦值來改變包含在委托變量中的方法地址引用。舊的引用會被垃圾回收器回收。

MyDel del;
del = myInstaObj.MyM1; //委托初始化
del = SClass.OtherM2;//委托重新賦值,舊的引用將被回收

6. 組合委托

委托可以使用額外的運算符來組合。這個運算最終會創建一個新的委托,其調用列表是兩個操作數的委托調用列表的副本的連接。

委托是恒定的,操作數委托創建後不會被改變。委托組合拷貝的是操作數的副本

MyDel del1 = myObj.MyMethod;
MyDel del2 = SClass.OtherM2;
MyDel del3 = del1 + del2;   //組合調用列表

7. 委托加減運算

可以使用+=運算符,為委托新增方法。

同樣可以使用-=運算符,為委托移除方法。

MyDel del = myObj.MyMethod;
del += SClass.OtherM2; // 增加方法
del -= myObj.MyMethod; // 移除方法

8. 委托調用

委托調用跟方法調用類似。委托調用後,調用列表的每個方法將會被執行。

在調用委托前,應判斷委托是否為空。調用空委托會拋出異常。

if(null != del)
{
     del();//委托調用
}

9. 匿名方法

匿名方法是在初始化委托時內聯聲明的方法。

基本結構:

deleage( 參數 ) { 語句塊 }

例如:


delegate int MyDel (int x); //定義一個委托 

MyDel del = delegate( int x){ return x; };

從上面我們可以看到,匿名方法是不會顯示聲明返回值的

10. Lambda表達式

Lambda表達式主要用來簡化匿名方法的語法。在匿名方法中,delegate關鍵字有點多余,因為編譯器已經知道我們將方法賦值給委托。通過幾個簡單步驟,我們就可以將匿名方法轉換為Lambda表達式:

  • 刪除delegate關鍵字
  • 在參數列表和匿名方法主體之間防Lambda運算符=>。Lambda運算符讀作"goes to"。
MyDel del = delegate( int x) { return x; };//匿名方法
MyDel del2 = (int x) => {return x;};//Lambda表達式
MyDel del3 = x => {return x};//簡寫的Lambda表達式

C#委托使用詳解(Delegates)