C#中委托的+=和-=深入研究
寫在前面
為什么會(huì)突然想說說委托?原因嗎,起于一個(gè)同事的想法,昨天下班的路上一直在想這個(gè)問題,如果給委托注冊(cè)多個(gè)方法,會(huì)不會(huì)都執(zhí)行呢?為了一探究性,就弄了個(gè)demo研究下。
+=
大家都知道委托都繼承自System.MulticastDelegate,而System.MulticastDelegate又繼承自System.Delegate,可以通過+=為委托注冊(cè)多個(gè)方法。那么他們是否都執(zhí)行了呢?執(zhí)行的結(jié)果又是怎樣的呢?有返回值和沒返回值的是否結(jié)果是否一樣?那就試著說說+=都干了哪些事?
測試代碼
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;
}
}
}
你可以猜猜運(yùn)行結(jié)果,如下圖:
可以看到?jīng)]有返回值的都輸出了,有返回值的只輸出了Mutiply的結(jié)果,那么+=內(nèi)部做了哪些事?可以看一下反編譯的代碼:
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;
}
}
}
通過上面的代碼可以看出+=內(nèi)部是通過委托的 Combine靜態(tài)方法將委托進(jìn)行組合的,可以看一下委托的這個(gè)靜態(tài)方法是如何實(shí)現(xiàn)的。
可以看到最終調(diào)用CombineImpl這個(gè)方法,這個(gè)方法內(nèi)部很奇怪:
并沒有我們想看到的代碼,那這個(gè)方法是干嘛用的?。?/p>
MSDN的解釋
Concatenates the invocation lists of the specified multicast (combinable) delegate and the current multicast (combinable) delegate.
大概意思就是:將當(dāng)前的委托加入到指定的多播委托集合中。
繞了一圈那么有返回值的委托,到底執(zhí)行了么?那也只能通過調(diào)試來看看了。(繞了一圈,又回到了編輯器,唉)
繼續(xù)F11你會(huì)發(fā)現(xiàn)確實(shí)進(jìn)入了Add方法
也確實(shí)執(zhí)行了,但在遍歷多播委托集合的時(shí)候,將之前的值給覆蓋了。
那么現(xiàn)在可以得出這樣的結(jié)論了:無返回值的委托,你給它注冊(cè)多少個(gè)方法,它就執(zhí)行多少個(gè)方法,而有返回值的委托,同樣注冊(cè)多少個(gè)方法就執(zhí)行多少個(gè)方法,但返回的是最后一個(gè)方法的返回值。
-=
既然說了+=,那么作為收拾爛攤子的-=也不得不說。在項(xiàng)目中使用了+=就要使用-=來釋放。那它內(nèi)部做了哪些事?同樣使用上面的代碼,在輸出結(jié)果后,使用-=來釋放資源。
可以看出,使用-=內(nèi)部是調(diào)用了委托的Remove靜態(tài)方法。
使用-=最終是將委托置為null,為null另一個(gè)意思就是空引用,這樣就可以等待垃圾回收器進(jìn)行回收了。
總結(jié)
這個(gè)問題雖然很基礎(chǔ),一個(gè)同事當(dāng)時(shí)問了,就給他說了一下,在下班的路上一直在想,內(nèi)部是如何實(shí)現(xiàn)的?就試著通過反編譯的方式一探究竟。但貌似CombineImpl這個(gè)方法,給的結(jié)果不太滿意。沒看到具體的實(shí)現(xiàn)。希望對(duì)你有所幫助!
相關(guān)文章
C# Winform中DataGridView導(dǎo)出為Excel的實(shí)現(xiàn)示例
本文主要介紹了C# Winform中DataGridView導(dǎo)出為Excel的實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05C# StreamReader類實(shí)現(xiàn)讀取文件的方法
這篇文章主要介紹了C# StreamReader類實(shí)現(xiàn)讀取文件的方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01C#實(shí)現(xiàn)汽車租賃系統(tǒng)項(xiàng)目
這篇文章主要為大家詳細(xì)介紹了C#實(shí)現(xiàn)汽車租賃系統(tǒng)項(xiàng)目,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01C#中使用IFormattable實(shí)現(xiàn)自定義格式化字符串輸出示例
這篇文章主要介紹了C#中使用IFormattable實(shí)現(xiàn)自定義格式字符串輸出示例,本文直接給出實(shí)例代碼,需要的朋友可以參考下2015-06-06c# 數(shù)據(jù)類型占用的字節(jié)數(shù)介紹
本篇文章主要是對(duì)c#中數(shù)據(jù)類型占用的字節(jié)數(shù)進(jìn)行了詳細(xì)的介紹。需要的朋友可以過來參考下,希望對(duì)大家有所幫助2014-01-01