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

詳解C#中的委托

 更新時(shí)間:2017年01月17日 09:16:37   作者:鄒瓊俊  
本文主要介紹了C#中委托的相關(guān)知識(shí)。具有一定的參考價(jià)值,下面跟著小編一起來(lái)看下吧

委托這個(gè)東西不是很好理解,可是工作中又經(jīng)常用到,你隨處可以看到它的身影,真讓人有一種又愛(ài)又恨的感覺(jué),我相信許多人被它所困擾過(guò)。

一提到委托,如果你學(xué)過(guò)C語(yǔ)言,你一定會(huì)馬上聯(lián)想到函數(shù)指針。

什么是委托?委托是C#中類(lèi)型安全的,可以訂閱一個(gè)或多個(gè)具有相同簽名方法的函數(shù)指針。委托可以把函數(shù)做為參數(shù)傳遞,其實(shí)際意義便是讓別人代理你的事情。委托可以看做是函數(shù)的指針,整數(shù)可以用整數(shù)變量指向它,對(duì)象可以用對(duì)象變量指向它,

函數(shù)也可以用委托變量指向它。我們可以選擇將委托類(lèi)型看做只定義了一個(gè)方法的接口,而委托的實(shí)例可以看做是實(shí)現(xiàn)了那個(gè)接口的一個(gè)對(duì)象。

使用委托,必須滿足4個(gè)條件:

  • 聲明委托類(lèi)型;
  • 必須有一個(gè)方法包含了要執(zhí)行的代碼;
  • 必須創(chuàng)建一個(gè)委托實(shí)例;
  • 必須調(diào)用(invoke)委托實(shí)例。

委托的申明

聲明委托的方式:delegate 返回值類(lèi)型 委托類(lèi)型名(參數(shù))

委托的申明和接口方法的申明基本上一致,只是在返回類(lèi)型關(guān)鍵字的前面多了一個(gè)delegate關(guān)鍵字。還有就是委托一般聲明為public類(lèi)型,因?yàn)樗S時(shí)要供別人調(diào)用的。

委托的本質(zhì)也是一個(gè)類(lèi)型。我們聲明一個(gè)類(lèi)可以進(jìn)行實(shí)例化,同樣委托也可以進(jìn)行實(shí)例化。

有如下四種委托:

//1.無(wú)參數(shù)無(wú)返回值
    public delegate void NoParaNoReturnEventHandler();
    //2.有參數(shù)無(wú)返回值
    public delegate void WithParaNoReturnEventHandler(string name);
    //3.無(wú)參數(shù)有返回值
    public delegate string NoParaWithReturnEventHandler();
    //4.有參數(shù)有返回值
    public delegate string WithParaWithReturnEventHandler(string name);

如果代碼想要執(zhí)行操作,但不知道操作細(xì)節(jié),一般可以使用委托。例如, Thread類(lèi)之所以知道要在一個(gè)新線程里運(yùn)行什么,唯一的原因就是在啟動(dòng)新線程時(shí),向它提供了一個(gè)ThreadStart或ParameterizedThreadStart委托實(shí)例。

Thread th = new Thread(Test);
th.Start();
public Thread(ThreadStart start);
public delegate void ThreadStart();

ThreadStart是一個(gè)無(wú)參無(wú)返回值的委托。

    static void Test()
    {
      Console.WriteLine("線程方法");
    }

這個(gè)Test方法的函數(shù)簽名必須和委托ThreadStart的函數(shù)簽名一致。

委托的調(diào)用

必須先實(shí)例化委托,然后再調(diào)用。

函數(shù)的簽名和委托的簽名必須一致。NoParaNoReturnEventHandler _NoParaNoReturnEventHandler = ConsoleInfo;,編譯器幫我們進(jìn)行了new,但是不能寫(xiě)成NoParaNoReturnEventHandler _NoParaNoReturnEventHandler = ConsoleInfo();

因?yàn)檫@樣就成為了函數(shù)調(diào)用。

#region 無(wú)返回值委托調(diào)用
    public static void Show()
    {
      //實(shí)例化委托
      NoParaNoReturnEventHandler _NoParaNoReturnEventHandler = new NoParaNoReturnEventHandler(ConsoleInfo);
      //NoParaNoReturnEventHandler _NoParaNoReturnEventHandler = ConsoleInfo; //簡(jiǎn)寫(xiě)
      //委托調(diào)用 通過(guò)Invoke()調(diào)用,或者可以直接省略
      _NoParaNoReturnEventHandler.Invoke();
      //_NoParaNoReturnEventHandler();
    }
    private static void ConsoleInfo()
    {
      Console.WriteLine("無(wú)參數(shù)無(wú)返回值的函數(shù)調(diào)用");
    } 
    #endregion

沒(méi)有委托就沒(méi)有異步,異步正是因?yàn)槲械拇嬖凇?/p>

_NoParaNoReturnEventHandler.BeginInvoke(null,null); //異步調(diào)用

為什么要使用委托

我們完全可以直接調(diào)用方法,為什么還需要通過(guò)一個(gè)委托來(lái)調(diào)用呢?委托有什么意義?

解耦,對(duì)修改關(guān)閉,對(duì)擴(kuò)展開(kāi)放。邏輯分離。

你可以把委托理解為函數(shù)的父類(lèi),或者是一個(gè)方法的占位符。

我們來(lái)看下代碼,假設(shè)有2個(gè)方法,一個(gè)說(shuō)英語(yǔ),一個(gè)說(shuō)漢語(yǔ),而這2個(gè)方法的函數(shù)簽名是一樣的。

public static void SayChinese(string name)
    {
      Console.WriteLine("你好," + name);
    }
    public static void SayEnglish(string name)
    {
      Console.WriteLine("hello," + name);
    }

那么我們?cè)谕獠空{(diào)用的時(shí)候,

  MyDelegate.SayChinese("張三");
  MyDelegate.SayEnglish("zhangsan");

如果要調(diào)用這兩個(gè)不同的方法,是不是要寫(xiě)不同的調(diào)用代碼

我們能不能只一個(gè)方法調(diào)用呢?修改代碼如下:

public static void Say(string name,WithParaNoReturnEventHandler handler)
    {
      handler(name);
    }
   public static void SayChinese(string name)
    {
      Console.WriteLine("你好," + name);
    }
    public static void SayEnglish(string name)
    {
      Console.WriteLine("hello," + name);
    }

這樣,只通過(guò)一個(gè)方法Say來(lái)進(jìn)行調(diào)用。

如何調(diào)用呢?如下三種調(diào)用方式:

      WithParaNoReturnEventHandler _WithParaNoReturnEventHandler = new WithParaNoReturnEventHandler(MyDelegate.SayChinese);
      MyDelegate.Say("張三",_WithParaNoReturnEventHandler);
      MyDelegate.Say("張三", delegate(string name) { Console.WriteLine("你好," + name); }); //匿名方法
      MyDelegate.Say("張三", (name) => { Console.WriteLine("你好," + name); }); //lambda表達(dá)式

以上代碼使用了幾種調(diào)用方式,這些調(diào)用方式都是隨著C#的升級(jí)而不斷優(yōu)化的。第一種是C#1.0中就存在的傳統(tǒng)調(diào)用方式,第二種是C#2.0中的匿名方法調(diào)用方式,所謂匿名方法,就是沒(méi)有名字的方法,當(dāng)方法只調(diào)用一次時(shí)使用匿名方法最合適不過(guò)了。C#3中的lambda表達(dá)式。其實(shí)泛型委托同樣是被支持的,而.NET 3.5則更進(jìn)一步,引入了一組名為Func的泛型委托類(lèi)型,它能獲取多個(gè)指定類(lèi)型的參數(shù),并返回另一個(gè)指定類(lèi)型的值。

lambda表達(dá)式

lambda表達(dá)式的本質(zhì)就是一個(gè)方法,一個(gè)匿名方法。

如果方法體只有一行,無(wú)返回值,還可以去掉大括號(hào)和分號(hào)。

MyDelegate.Say("張三", (name) => Console.WriteLine("你好," + name));

如果方法體只有一行,有返回值,可以去掉大括號(hào)和return。

WithParaWithReturnEventHandler _WithParaWithReturnEventHandler = (name)=>name+",你好";

從.NET3.5開(kāi)始,基本上不需要我們自己來(lái)申明委托了,因?yàn)橄到y(tǒng)有許多內(nèi)置的委托。

Action和Func委托,分別有16個(gè)和17個(gè)重載。int表示輸入?yún)?shù),out代表返回值,out參數(shù)放置在最后。

Action表示無(wú)返回值的委托,F(xiàn)unc表示有返回值的委托。因?yàn)榉椒◤拇蟮慕嵌葋?lái)分類(lèi),也分為有返回值的方法和無(wú)返回值的方法。

也就是說(shuō)具體調(diào)用什么樣的方法,完全由調(diào)用方?jīng)Q定了,就有了更大的靈活性和擴(kuò)展性。為什么這么說(shuō),如果我有些時(shí)候要先說(shuō)英語(yǔ)再說(shuō)漢語(yǔ),有些事時(shí)候要先說(shuō)漢語(yǔ)再說(shuō)英語(yǔ),如果沒(méi)有委托,我們會(huì)怎么樣實(shí)現(xiàn)?請(qǐng)看如下代碼:

public static void SayEnglishAndChinese(string name)
    {
      SayEnglish(name);
      SayChinese(name);
    }
    public static void SayChineseAndEnglish(string name)
    {
      SayChinese(name);
      SayEnglish(name);
    }

如果又突然要添加一種俄語(yǔ)呢?被調(diào)用方的代碼又要修改,如此循環(huán)下去,是不是要抓狂了?隨著不斷添加新語(yǔ)種,代碼會(huì)變得越來(lái)越復(fù)雜,越來(lái)越難以維護(hù)。這樣的代碼耦合性非常高,是不合理的,也就是出現(xiàn)了所謂的代碼的壞味道,你可以通過(guò)設(shè)計(jì)模式(如觀察者模式等),在不使用委托的情況下來(lái)重構(gòu)代碼,但是實(shí)現(xiàn)起來(lái)是非常麻煩的,要寫(xiě)很多更多的代碼...

委托可以傳遞方法,而這些方法可以代表一系列的操作,這些操作都由調(diào)用方來(lái)決定,就很好擴(kuò)展了,而且十分靈活。我們不會(huì)對(duì)已有的方法進(jìn)行修改,而是只以添加方法的形式去進(jìn)行擴(kuò)展。

可能有人又會(huì)說(shuō),我直接在調(diào)用方那里來(lái)一個(gè)一個(gè)調(diào)用我要執(zhí)行哪些方法一樣可以實(shí)現(xiàn)這樣的效果???

可你有沒(méi)有想過(guò),你要調(diào)用的是一系列方法,你根本無(wú)法復(fù)用這一系列的方法。使用委托就不一樣了,它好比一個(gè)方法集合的容器,你可以往里面增減方法,可以復(fù)用的。而且使用委托,你可以延時(shí)方法列表的調(diào)用,還可以隨時(shí)對(duì)方法列表進(jìn)行增減。委托對(duì)方法進(jìn)行了再一次的封裝。

總結(jié):也就是當(dāng)你只能確定方法的函數(shù)簽名,無(wú)法確定方法的具體執(zhí)行時(shí),為了能夠更好的擴(kuò)展,以類(lèi)似于注入方法的形式來(lái)實(shí)現(xiàn)新增的功能,就能體現(xiàn)出委托的價(jià)值。

委托和直接調(diào)用函數(shù)的區(qū)別:用委托就可以指向任意的函數(shù),哪怕是之前沒(méi)定義的都可以,而不用受限于哪幾種。

多播委托

組合的委托必須是同一個(gè)類(lèi)型,其相當(dāng)于創(chuàng)建了一個(gè)按照組合的順序依次調(diào)用的新委托對(duì)象。委托的組合一般是給事件用的,用普通委托的時(shí)候很少用。

通過(guò)+來(lái)實(shí)現(xiàn)將方法添加到委托實(shí)例中,-來(lái)從委托實(shí)例中進(jìn)行方法的移除。

+和-純粹是為了簡(jiǎn)化代碼而生的,實(shí)際上其調(diào)用的分別是Delegate.Combine方法和Delegate.Remove。

如果委托中存在多個(gè)帶返回值的方法,那么調(diào)用委托的返回值是最后一個(gè)方法的返回值。

public static void MultipleShow()
    {
      //多播委托
      NoParaWithReturnEventHandler _NoParaWithReturnEventHandler = new NoParaWithReturnEventHandler(GetDateTime);
      _NoParaWithReturnEventHandler += GetDateTime;
      Console.WriteLine(_NoParaWithReturnEventHandler());
    }
    public static string GetDateTime()
    {
      return string.Format("今天是{0}號(hào)。", DateTime.Now.Day.ToString());
    }

委托總結(jié):

  • 委托封裝了包含特殊返回類(lèi)型和一組參數(shù)的行為,類(lèi)似包含單一方法的接口;
  • 委托類(lèi)型聲明中所描述的類(lèi)型簽名決定了哪個(gè)方法可用于創(chuàng)建委托實(shí)例,同時(shí)決定了調(diào)用的簽名;
  • 為了創(chuàng)建委托實(shí)例,需要一個(gè)方法以及(對(duì)于實(shí)例方法來(lái)說(shuō))調(diào)用方法的目標(biāo);
  • 委托實(shí)例是不易變的,就像String一樣;
  • 每個(gè)委托實(shí)例都包含一個(gè)調(diào)用列表——一個(gè)操作列表;
  • 事件不是委托實(shí)例——只是成對(duì)的add/remove方法(類(lèi)似于屬性的取值方法/賦值方法)。

常見(jiàn)使用場(chǎng)景:窗體傳值、線程啟動(dòng)時(shí)綁定方法、lambda表達(dá)式、異步等等。

生活中的例子:現(xiàn)在不是大家都在搶火車(chē)票嗎,使用云搶票就相當(dāng)于使用委托,你可以直接自己買(mǎi)票,也可以托管于云搶票,自己搶票的話,在快要開(kāi)槍的時(shí)候,你必須時(shí)刻刷新,下單輸驗(yàn)證碼等等,使用云搶票的話,你只要放票前,提前輸入搶票信息,就再也不需要你管了,自動(dòng)出票,你根本不需要知道云搶票那邊是怎么幫你實(shí)現(xiàn)搶票的。相同時(shí)間和車(chē)次可以做成一個(gè)委托實(shí)例,有很多人都通過(guò)這個(gè)委托實(shí)例來(lái)進(jìn)行搶票操作。

源碼下載:http://pan.baidu.com/s/1dEPlxJj

以上就是本文的全部?jī)?nèi)容,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,同時(shí)也希望多多支持腳本之家!

相關(guān)文章

最新評(píng)論