首頁  >  文章  >  後端開發  >  深入講解C#中委託的+=和-=

深入講解C#中委託的+=和-=

Y2J
Y2J原創
2017-04-21 14:15:254220瀏覽

這篇文章主要介紹了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中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn