這篇文章主要介紹了C#中委託的+=和-=深入研究,本文深入研究+=和-=在執行時都做了哪些事情,加深對C#委託的理解和使用,需要的朋友可以參考下
寫在前面
為什麼會突然想說說委託?原因嗎,起於一個同事的想法,昨天下班的路上一直在想這個問題,如果給委託註冊多個方法,會不會都執行呢?為了一探究性,就弄了demo研究下。
+=
大家都知道委託都繼承自System.MulticastDelegate,而System.MulticastDelegate又繼承自System.Delegate,可以透過+=為委託註冊多個方法。那麼他們是否都執行了呢?執行的結果又是怎麼樣的呢?有回傳值和沒回傳值的是否結果是否一樣?那就試著說+=都做了哪些事?
測試程式碼
程式碼如下:
namespace Wolfy.DelegateDemo { public delegate void ShowMsg(string msg); public delegate int MathOperation(int a, int b); class Program { static ShowMsg showMsg; static MathOperation mathOperation; static void Main(string[] args) { showMsg += ShowHello; showMsg += ShowHello1; showMsg("大家新年好啊"); mathOperation += Add; mathOperation += Multiply; int result = mathOperation(1, 2); Console.WriteLine(result.ToString()); Console.Read(); } static void ShowHello(string msg) { Console.WriteLine("哈喽:" + msg); } static void ShowHello1(string msg) { Console.WriteLine("哈喽1:" + msg); } static int Add(int a, int b) { return a + b; } static int Multiply(int a, int b) { return a * b; } } }
你可以猜猜看運行結果,如下圖:
可以看到沒有回傳值的都輸出了,有回傳值的只輸出了Mutiply的結果,那麼+=內部做了哪些事?可以看一下反編譯的程式碼:
程式碼如下:
using System; namespace Wolfy.DelegateDemo { internal class Program { private static ShowMsg showMsg; private static MathOperation mathOperation; private static void Main(string[] args) { Program.showMsg = (ShowMsg)Delegate.Combine(Program.showMsg, new ShowMsg(Program.ShowHello)); Program.showMsg = (ShowMsg)Delegate.Combine(Program.showMsg, new ShowMsg(Program.ShowHello1)); Program.showMsg("大家新年好啊"); Program.mathOperation = (MathOperation)Delegate.Combine(Program.mathOperation, new MathOperation(Program.Add)); Program.mathOperation = (MathOperation)Delegate.Combine(Program.mathOperation, new MathOperation(Program.Multiply)); Console.WriteLine(Program.mathOperation(1, 2).ToString()); Console.Read(); } private static void ShowHello(string msg) { Console.WriteLine("哈喽:" + msg); } private static void ShowHello1(string msg) { Console.WriteLine("哈喽1:" + msg); } private static int Add(int a, int b) { return a + b; } private static int Multiply(int a, int b) { return a * b; } } }
透過上面的程式碼可以看出+=內部是透過委託的Combine靜態方法將委託進行組合的,可以看一下委託的這個靜態方法是如何實現的。
可以看到最終呼叫CombineImpl這個方法,這個方法內部很奇怪:
沒有我們想看到的程式碼,那這個方法是幹嘛用的啊?
MSDN的解釋
Concatenates the invocation lists of the specified multicast (combinable) delegate and the current multicast (combinable) delegate.
#大概是:將目前的委託加入指定的多播委託集合中。
繞了一圈那麼有回傳值的委託,到底執行了麼?那也只能透過調試來看看了。 (繞了一圈,又回到了編輯器,唉)
繼續F11你會發現確實進入了Add方法
也確實執行了,但在遍歷多播委託集合的時候,將先前的值覆蓋了。
那麼現在可以得到這樣的結論了:無回傳值的委託,你給它註冊多少個方法,它就執行多少個方法,而有回傳值的委託,同樣註冊多少個方法就執行多少個方法,但回傳的是最後一個方法的回傳值。
-=
既然說了+=,那麼作為收拾爛攤子的-=也不得不說。在專案中使用了+=就要使用-=來釋放。那它內部做了哪些事?同樣使用上面的程式碼,在輸出結果後,使用-=來釋放資源。
可以看出,使用-=內部是呼叫了委託的Remove靜態方法。
使用-=最終是將委託置為null,為null另一個意思就是空引用,這樣就可以等待垃圾回收器進行回收了。
總結
這個問題雖然很基礎,一個同事當時問了,就給他說了一下,在下班的路上一直在想,內部是如何實現的?就試著透過反編譯的方式一探究竟。但似乎CombineImpl這個方法,給的結果不太滿意。沒看到具體的實現。希望對你有幫助!
以上是深入講解C#中委託的+=和-=的詳細內容。更多資訊請關注PHP中文網其他相關文章!