歡迎您光臨本站 註冊首頁

詳解C#之委託

←手機掃碼閱讀     kyec555 @ 2020-06-10 , reply:0

委託:顧名思義,讓別人幫你辦件事。委託是C#實現回撥函式的一種機制。可能有人會問了,回撥函式是個啥???

舉個例子:我現在是一家公司的老闆,公司現在在招聘.NET工程師,我們有一個小姐姐專門負責接受求職者投遞的簡歷,我就告訴這個小姐姐,一旦收到新的簡歷就轉發給我一份。

這個例子裡小姐姐要做的工作:給我轉發一份簡歷(回撥函式裡的操作),就是一個回撥函式的作用。一旦有了滿足條件(收到了新的簡歷),小姐姐就會轉發給我(觸發回撥函式)

用來程式碼來看看是怎麼實現的:

1.定義一個委託:

  // 定義委託,這個委託需要獲取一個int型引數,返回void    internal delegate void Feedback(int value);

 

2.定義回撥方法:

這裡定義了兩個方法,一個靜態,一個例項。正好看看呼叫方式的不同。注意:定義的回撥方法簽名必須和委託物件一致(這裡都是int 型別引數,沒有返回值。這麼說也不全對,涉及到協變和逆變。這裡就不解釋這倆了),這是因為將方法繫結到委託時,編譯器會檢測他們的相容性。不符合的話回報編譯錯誤。就比如有一個方法要傳入String型別,我們給它傳遞了一個int型別一樣。

這裡為了方便演示就只把數字列印在了控制檯。

  ////// 靜態回撥方法    //////private static void FeedbackToConsole(int value)    {     // 依次列印數字     Console.WriteLine("Item=" + value);    }    ////// 例項回撥方法    //////private void InstanceFeedbackToConsole(int value)    {     Console.WriteLine("Item=" + value);    }

 

3.編寫一個方法來觸發回撥函式:

有三個引數:前兩個做迴圈使用,後一個接收定義的委託物件。內部程式碼迴圈呼叫回撥方法 fb(val)的寫法,其實就是相當於要呼叫的函式。例:

FeedbackToConsole(val)

  ////// 使用此方法觸發委託回撥    //////開始///結束///委託引用private static void Counter(int from,int to, Feedback fb)    {     for (int val = from; val <= to; val++)     {      // fb不為空,則呼叫回撥方法      if (fb != null)      {       fb(val);      }      //fb?.Invoke(val); 簡化版本呼叫     }    }

 

4.定義Counter的方法呼叫(這一步可有可無,為了區分靜態和例項方法就寫了)

第一次呼叫Counter,傳遞Null,在回撥方法裡有一步判空操作,所以是不回撥用回撥函式的。第二個Counter呼叫正常傳遞引數,構造一個委託物件並綁定了一個方法

  ////// 靜態呼叫    ///private static void StaticDelegateDemo()    {     Console.WriteLine("---------委託呼叫靜態方法------------");     Counter(1, 10, null);     Counter(1, 10, new Feedback(FeedbackToConsole));                }      ////// 例項呼叫    ///private static void InstanceDelegateDemo()    {     Console.WriteLine("---------委託呼叫例項方法------------");     Program p = new Program();     Counter(1, 10, null);     Counter(1, 5, new Feedback(p.InstanceFeedbackToConsole));    }

 

5. 檢視控制檯資訊

完整程式碼:

  class Program   {    // 定義委託,並引用一個方法,這個方法需要獲取一個int型引數返回void    internal delegate void Feedback(int value);    static void Main(string[] args)    {          StaticDelegateDemo();     InstanceDelegateDemo();     Console.ReadKey();    }        ////// 靜態呼叫    ///private static void StaticDelegateDemo()    {     Console.WriteLine("---------委託呼叫靜態方法------------");     Counter(1, 10, null);     Counter(1, 10, new Feedback(FeedbackToConsole));           }      ////// 例項呼叫    ///private static void InstanceDelegateDemo()    {     Console.WriteLine("---------委託呼叫例項方法------------");     Program p = new Program();     Counter(1, 10, null);     Counter(1, 5, new Feedback(p.InstanceFeedbackToConsole));    }        ////// 靜態回撥方法    //////private static void FeedbackToConsole(int value)    {     // 依次列印數字     Console.WriteLine("Item=" + value);    }    ////// 例項回撥方法    //////private void InstanceFeedbackToConsole(int value)    {     Console.WriteLine("Item=" + value);    }   }

 

啟動控制檯:可以看到已經成功把數字列印出來了

6. 委託鏈:

委託鏈是委託物件的集合。可以利用委託鏈呼叫集合中的委託所繫結的全部方法。繼續在原有的基礎上新增委託鏈的方法。

新新增的兩個方法本質上沒有區別都是對委託鏈的實現,不同的是寫法,明顯是第二個方法更加精簡一些。這是因為C#編譯器過載了+=和-=運運算元,這兩個運運算元分別呼叫Combine和Remove。

  ////// 委託鏈呼叫 1    //////private static void ChainDelegateDemo(Program p)    {     Console.WriteLine("---------委託鏈呼叫1------------");     Feedback fb1 = new Feedback(FeedbackToConsole);     Feedback fb2 = new Feedback(p.InstanceFeedbackToConsole);     Feedback fbChain = null;     fbChain = (Feedback)Delegate.Combine(fbChain, fb1);     fbChain = (Feedback)Delegate.Combine(fbChain, fb2);     Counter(1, 3, fbChain);     Console.WriteLine();     fbChain = (Feedback)Delegate.Remove(fbChain, new Feedback(FeedbackToConsole));     Counter(1, 3, fbChain);    }      ////// 委託鏈呼叫 2    //////private static void ChainDelegateDemo2(Program p)    {     Console.WriteLine("---------委託鏈呼叫2------------");     Feedback fb1 = new Feedback(FeedbackToConsole);     Feedback fb2 = new Feedback(p.InstanceFeedbackToConsole);     Feedback fbChain = null;     fbChain += fb1;     fbChain += fb2;     Counter(1, 3, fbChain);     Console.WriteLine();     fbChain -= new Feedback(FeedbackToConsole);     Counter(1, 2, fbChain);    }

 

在Main方法新增對委託鏈的呼叫:

  static void Main(string[] args)    {     Program p = new Program();     StaticDelegateDemo();     InstanceDelegateDemo();     ChainDelegateDemo(p);     ChainDelegateDemo2(p);     Console.WriteLine("Hello World!");     Console.ReadKey();    }

 

啟動專案:

7. C#為委託提供的簡化:

7.1 不需要構造委託物件:

之前的程式碼:

Counter(1, 10, new Feedback(FeedbackToConsole));

構造了一個委託物件並傳遞給Counter方法,由於C#編譯器能自己推斷。所以可以省略構造委託物件,直接傳遞方法。使程式碼的可讀性更佳,也更容易理解。

簡化後的程式碼:

  ////// 靜態呼叫    ///private static void StaticDelegateDemo()    {     Console.WriteLine("---------委託呼叫靜態方法------------");     Counter(1, 10, null);     //Counter(1, 10, new Feedback(FeedbackToConsole));     Counter(1, 10, FeedbackToConsole);           }

 

可以看到效果是一樣的:

7.2 簡化語法:不需要定義回撥方法(以lambda表示式實現)

在前面的程式碼中定義了一個回撥方法:

  ////// 靜態回撥方法    //////private static void FeedbackToConsole(int value)    {     // 依次列印數字     Console.WriteLine("Item=" + value);    }

 

現在以lambda表示式方式實現:

  ////// 靜態呼叫    ///private static void StaticDelegateDemo()    {     Console.WriteLine("---------委託呼叫靜態方法------------");     Counter(1, 10, null);     //Counter(1, 10, new Feedback(FeedbackToConsole));     //Counter(1, 10, FeedbackToConsole);     Counter(1, 10, value => Console.WriteLine(value));      }

 

lambda表示式實際上是一個匿名函式。編譯器在看到lambda之後會在類中自動定義一個新的私有方法。類似於之前寫的回撥方法FeedbackToConsole()lambda必須匹配委託!

lambda的語法: 引數 => 方法體。

=>左邊是要傳入的引數,本例中是傳入一個Int型別的變數,=>右邊是具體的程式碼,相當於FeedbackToConsole(),{}中所做的操作

一些規則:

如果不傳遞引數: ()=>Console.WriteLine("Hello World!")

傳遞一個引數:(int n)=>Console.WriteLine(n.ToString())    或者去掉()和int  編譯器會自己推斷型別:n=>Console.WriteLine(n.ToString())

傳遞多個引數:(int n ,int m)=>Console.WriteLine(n.ToString())  或者編譯器自己推斷型別:(n , m)=>Console.WriteLine(n.ToString())

註:如果有一個方法需要多處呼叫或者方法裡面的程式碼量較多。還是單獨寫一個方法較為理想。

最後看一下換成lambda的寫法結果顯示是否一樣

全部程式碼:

  class Program   {    // 定義委託,並引用一個方法,這個方法需要獲取一個int型引數返回void    internal delegate void Feedback(int value);    static void Main(string[] args)    {     Program p = new Program();     StaticDelegateDemo();     InstanceDelegateDemo();     ChainDelegateDemo(p);     ChainDelegateDemo2(p);     Console.WriteLine("Hello World!");     string[] names = { "Jeff", "Jee", "aa", "bb" };     //char find = 'e';     //names= Array.FindAll(names, name => name.IndexOf(find) >= 0);     //Array.ForEach(names, Console.WriteLine);     Console.ReadKey();    }        ////// 靜態呼叫    ///private static void StaticDelegateDemo()    {     Console.WriteLine("---------委託呼叫靜態方法------------");     Counter(1, 10, null);     //Counter(1, 10, new Feedback(FeedbackToConsole));     //Counter(1, 10, FeedbackToConsole);     Counter(1, 10, value => Console.WriteLine(value));      }      ////// 例項呼叫    ///private static void InstanceDelegateDemo()    {     Console.WriteLine("---------委託呼叫例項方法------------");     Program p = new Program();     Counter(1, 10, null);     Counter(1, 5, new Feedback(p.InstanceFeedbackToConsole));    }      ////// 委託鏈呼叫 1    //////private static void ChainDelegateDemo(Program p)    {     Console.WriteLine("---------委託鏈呼叫1------------");     Feedback fb1 = new Feedback(FeedbackToConsole);     Feedback fb2 = new Feedback(p.InstanceFeedbackToConsole);     Feedback fbChain = null;     fbChain = (Feedback)Delegate.Combine(fbChain, fb1);     fbChain = (Feedback)Delegate.Combine(fbChain, fb2);     Counter(1, 3, fbChain);     Console.WriteLine();     fbChain = (Feedback)Delegate.Remove(fbChain, new Feedback(FeedbackToConsole));     Counter(1, 3, fbChain);    }      ////// 委託鏈呼叫 2    //////private static void ChainDelegateDemo2(Program p)    {     Console.WriteLine("---------委託鏈呼叫2------------");     Feedback fb1 = new Feedback(FeedbackToConsole);     Feedback fb2 = new Feedback(p.InstanceFeedbackToConsole);     Feedback fbChain = null;     fbChain += fb1;     fbChain += fb2;     Counter(1, 3, fbChain);     Console.WriteLine();     fbChain -= new Feedback(FeedbackToConsole);     Counter(1, 2, fbChain);    }    ////// 使用此方法觸發委託回撥    //////開始///結束///委託引用private static void Counter(int from,int to, Feedback fb)    {     for (int val = from; val <= to; val++)     {      // fb不為空,則呼叫回撥方法      if (fb != null)      {       fb(val);      }      //fb?.Invoke(val); 簡化版本呼叫     }    }      ////// 靜態回撥方法    //////private static void FeedbackToConsole(int value)    {     // 依次列印數字     Console.WriteLine("Item=" + value);    }    ////// 例項回撥方法    //////private void InstanceFeedbackToConsole(int value)    {     Console.WriteLine("Item=" + value);    }   }

                  

   


[kyec555 ] 詳解C#之委託已經有380次圍觀

http://coctec.com/docs/vscode/show-post-237896.html