欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C#中委托的進(jìn)一步理解

 更新時(shí)間:2015年02月26日 10:09:20   投稿:junjie  
這篇文章主要介紹了C#中委托的進(jìn)一步理解,本文講解了委托類型、建立委托鏈、移除委托鏈等內(nèi)容,需要的朋友可以參考下

前面一篇文章介紹了委托的基本知識(shí),接下來就進(jìn)一步研究一下委托。

委托類型

其實(shí),剛開始覺得委托類型是一個(gè)比較難理解的概念,怎么也不覺得下面的"AssembleIphoneHandler"是一個(gè)類型。

復(fù)制代碼 代碼如下:

public delegate void AssembleIphoneHandler();

按照正常的情況,如果我們要?jiǎng)?chuàng)建一個(gè)委托類型應(yīng)該是:
復(fù)制代碼 代碼如下:

public class AssembleIphoneHandler : System.MulticastDelegate
{
}

但是,這種寫法是編譯不過的,會(huì)提示不能從"System.MulticastDelegate"派生子類。

其實(shí),這里是編譯器為我們做了一個(gè)轉(zhuǎn)換,當(dāng)我們使用delegate關(guān)鍵字聲明一個(gè)委托類型的時(shí)候,編譯器就會(huì)按照上面代碼片段中的方式為我們創(chuàng)建一個(gè)委托類型。

知道了這些東西,對(duì)于委托類型的理解就比較容易了,通過delegate聲明的委托類型就是一個(gè)從"System.MulticastDelegate"派生出來的子類。

建立委托鏈

下面我們通過一個(gè)例子來看看委托鏈的建立,以及調(diào)用列表的變化,基于前面一篇文章中的例子進(jìn)行一些修改。

復(fù)制代碼 代碼如下:

class Program
{
    static void Main(string[] args)
    {
        Apple apple = new Apple();
        Foxconn foxconn = new Foxconn();

        Apple.AssembleIphoneHandler d1, d2, d3, d4 = null;
        d1 = new Apple.AssembleIphoneHandler(foxconn.AssembleIphone);
        d2 = new Apple.AssembleIphoneHandler(foxconn.PackIphone);
        d3 = new Apple.AssembleIphoneHandler(foxconn.ShipIphone);

        d4 += d1;
        d4 += d2;
        d4 += d3;

        d4();

        Console.Read();
    }
}

我們接下來進(jìn)行一下單步調(diào)試看看委托鏈建立的過程。

1. 當(dāng)下面三句執(zhí)行完成后,可以通過VS看到d1、d2和d3的詳細(xì)信息

復(fù)制代碼 代碼如下:

d1 = new Apple.AssembleIphoneHandler(foxconn.AssembleIphone);

復(fù)制代碼 代碼如下:

d2 = new Apple.AssembleIphoneHandler(foxconn.PackIphone);

復(fù)制代碼 代碼如下:

d3 = new Apple.AssembleIphoneHandler(foxconn.ShipIphone);

對(duì)于上面三個(gè)委托實(shí)例來說:

1.調(diào)用列表為空,所以_invocationCount為0,_invocationList為空
2._target代表創(chuàng)建委托實(shí)例的方法來自Foxconn的實(shí)例;如果是靜態(tài)方法創(chuàng)建的委托實(shí)例_target值為null
3._methodPtr代表這個(gè)方法的唯一標(biāo)識(shí),可以理解為句柄
4._methodBase包含創(chuàng)建委托實(shí)例的方法的信息,方法名、返回類型等等

2. 通過"+="操作符來進(jìn)行委托合并

復(fù)制代碼 代碼如下:

d4 += d1;

這時(shí),由于d4初始值為null,在使用"+="操作(Combine方法)構(gòu)造委托鏈時(shí),將返回另外一個(gè)參數(shù)d1,再將d1的引用賦給d4(通過"ILSpy"查看,如下圖)。也就是說,這時(shí)d4將指向d1所指向的對(duì)象。

3. 繼續(xù)執(zhí)行委托合并,并查看d4的變化

復(fù)制代碼 代碼如下:

d4 += d2;

這時(shí)可以看到調(diào)用列表的變化,_invocationList包含兩個(gè)元素,分別是d1和d2.

4. 最后進(jìn)行一次委托合并,把d3合并到d4中

復(fù)制代碼 代碼如下:

d4 += d3;

可以看到最新的d4實(shí)例中,調(diào)用列表已經(jīng)包含了d3。

注意:由于委托是不可變的,所以這里應(yīng)該描述為,d3和d4的Combine 產(chǎn)生了一個(gè)新的委托實(shí)例,新的委托實(shí)例的調(diào)用列表是d3和d4的合并;操作完成后,d4變量將指向新的委托實(shí)例的引用。

疑問:其實(shí)在這步調(diào)試過程中有個(gè)疑問,_invocationCount的值是3,但是_invocationList中有四個(gè)元素,最后一個(gè)為null,找了一下也沒發(fā)現(xiàn)為什么,望高手看到幫忙解答。

所以對(duì)委托鏈建立的方法Delegate.Combine(Delegate A, Delegate B),可以進(jìn)行下面的概括:

1.如果A和B均為null,則返回null。
2.如果A或B一個(gè)為null而另一個(gè)不為null,則返回不為null的委托。
3.如果A和B均不為null,返回一個(gè)新的委托(委托是不可變的),該委托_invocationList字段為一個(gè)委托數(shù)組,該數(shù)組中委托的順序?yàn)椋篈中_invacationList所指向的委托數(shù)組 + B中_invacationList所指向的委托數(shù)組。

移除委托鏈

我們可以通過Delegate類的靜態(tài)方法Remove,從一個(gè)委托鏈中移除一個(gè)委托,這里就不做演示了。

注意:當(dāng)調(diào)用Remove時(shí),會(huì)遍歷(倒序)第一個(gè)參數(shù)中的中的調(diào)用列表(_invocationList), 找到與第二個(gè)參數(shù)的_target和_methodPtr字段相匹配的委托,并將其從委托列表中移除。

當(dāng)有多個(gè)匹配的情況是,Remove方法只移除第一個(gè)匹配的委托;但是,可以通過RemoeAll方法來移除所有匹配的委托。

同樣對(duì)委托移除的方法Delegate.Remove(Delegate A, Delegate B),可以進(jìn)行下面的概括:

1.如果A為null,返回null。
2.如果B為null,返回A。
3.如果A的_invocationList為null,即不包含委托鏈,那么如果A本身與B匹配,則返回null,否則返回A。
4.如果A的_invocationList中不包含與B匹配的委托,則返回A。
5.如果A的_invocationList中包含與B匹配的委托,則從鏈表中移除B,然后
6.如果A的鏈表中只剩下一個(gè)委托,則返回該委托。
6-1).如果A的鏈表中還剩下多個(gè)委托,將重新構(gòu)建一個(gè)新的委托,并且新的委托的_invocationList為A的6-2)._invocationList移除了B之后的List。

總結(jié)

通過這篇文章,進(jìn)一步認(rèn)識(shí)了委托類型,然后通過一個(gè)例子觀察了委托鏈的建立以及調(diào)用列表的變化。

通過這兩篇文章,對(duì)委托應(yīng)該有了一定的認(rèn)識(shí):

1.通過delegate關(guān)鍵字聲明委托類型

復(fù)制代碼 代碼如下:

[<修飾符>] delegate <返回類型> <委托名> ([<形參表>])

2.找到與委托簽名相符的方法來創(chuàng)建委托實(shí)例,也可以通過"+="和"-="來組合和移除委托
復(fù)制代碼 代碼如下:

new <委托類型名> (<方法>)

3.通過委托實(shí)例調(diào)用委托

相關(guān)文章

  • C# List<T>的用法小結(jié)

    C# List<T>的用法小結(jié)

    本篇文章主要是對(duì)C#中List<T>的用法進(jìn)行了詳細(xì)的總結(jié)介紹,需要的朋友可以過來參考下,希望對(duì)大家有所幫助
    2014-01-01
  • C#實(shí)現(xiàn)十字鏈表的使用示例

    C#實(shí)現(xiàn)十字鏈表的使用示例

    十字鏈表是一種將數(shù)據(jù)存儲(chǔ)在節(jié)點(diǎn)中的數(shù)據(jù)結(jié)構(gòu),每個(gè)節(jié)點(diǎn)包含兩個(gè)指針,分別指向下一個(gè)節(jié)點(diǎn)和上一個(gè)節(jié)點(diǎn),通過定義節(jié)點(diǎn)類和鏈表類,實(shí)現(xiàn)十字鏈表的創(chuàng)建、遍歷、插入和刪除等操作,本文就來實(shí)現(xiàn)一下
    2023-11-11
  • c#如何實(shí)現(xiàn)接口事件

    c#如何實(shí)現(xiàn)接口事件

    這篇文章主要介紹了c#如何實(shí)現(xiàn)接口事件,幫助大家更好的理解和學(xué)習(xí)c#,感興趣的朋友可以了解下
    2020-10-10
  • C#最小二乘法擬合曲線成直線的實(shí)例

    C#最小二乘法擬合曲線成直線的實(shí)例

    這篇文章主要介紹了C#最小二乘法擬合曲線成直線的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-02-02
  • c#?復(fù)寫Equals方法的實(shí)現(xiàn)

    c#?復(fù)寫Equals方法的實(shí)現(xiàn)

    本文主要介紹了c#?復(fù)寫Equals方法的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-05-05
  • C#調(diào)用Windows的API實(shí)現(xiàn)窗體動(dòng)畫

    C#調(diào)用Windows的API實(shí)現(xiàn)窗體動(dòng)畫

    在VF、VB、PB的應(yīng)用中,有些無法通過語言工具本身來完成的或者做得不理想的功能,我們會(huì)考慮通過Windows的API來完成。本文就來通過調(diào)用Windows的API實(shí)現(xiàn)窗體動(dòng)畫,感興趣的可以嘗試一下
    2022-11-11
  • C#數(shù)據(jù)庫操作類AccessHelper實(shí)例

    C#數(shù)據(jù)庫操作類AccessHelper實(shí)例

    這篇文章主要介紹了C#數(shù)據(jù)庫操作類AccessHelper實(shí)例,可實(shí)現(xiàn)針對(duì)access數(shù)據(jù)庫的各種常見操作,非常具有實(shí)用價(jià)值,需要的朋友可以參考下
    2014-10-10
  • c#中SqlHelper封裝SqlDataReader的方法

    c#中SqlHelper封裝SqlDataReader的方法

    這篇文章主要介紹了c#中SqlHelper封裝SqlDataReader的方法,涉及C#針對(duì)數(shù)據(jù)庫相關(guān)操作封裝與使用的技巧,需要的朋友可以參考下
    2015-05-05
  • C#?WPF后臺(tái)動(dòng)態(tài)添加控件實(shí)戰(zhàn)教程

    C#?WPF后臺(tái)動(dòng)態(tài)添加控件實(shí)戰(zhàn)教程

    最近嘗試用wpf在后臺(tái)動(dòng)態(tài)添加控件,所以下面這篇文章主要給大家介紹了關(guān)于C#?WPF后臺(tái)動(dòng)態(tài)添加控件的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-05-05
  • C#中Try-Catch語句真的影響程序性能嗎?

    C#中Try-Catch語句真的影響程序性能嗎?

    這篇文章主要介紹了C#中Try-Catch語句真的影響程序性能嗎?本文結(jié)合IL分析Try-Catch語句的性能問題,需要的朋友可以參考下
    2015-06-06

最新評(píng)論