C# 委託使用詳解(delegate)
阿新 • • 發佈:2019-01-02
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表示式