1. 委託是什嗎?
其實,我一直思考如何講解委託,才能把委託說得更透徹。說實話,每個人都委託都有不同的見解,因為看問題的角度不同。個人認為,可以從以下2點來理解:
(1) 從資料結構來講,委託是和類一樣是一種使用者自訂類型。
(2) 從設計模式來講,委託(類)提供了方法(對象)的抽象。
既然委託是一種類型,那麼它儲存的是什麼資料?
我們知道,委託是方法的抽象,它儲存的就是一系列具有相同簽名和返回回類型的方法的地址。調用委託的時候,委託包含的所有方法將被執行。
2. 委託類型的定義
委託是類型,就好像類是類型一樣。與類一樣,委託類型必須在被用來建立變數以及類型對象之前聲明。
delegate void MyDel(int x);
委託型別宣告:
(1) 以deleagate關鍵字開頭。
(2)傳回型別+委託類型名+參數列表。
delegate void MyDel(int x);
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運算式