溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

C#中委托和匿名委托有什么用

發(fā)布時間:2021-03-06 12:45:51 來源:億速云 閱讀:147 作者:小新 欄目:編程語言

這篇文章將為大家詳細(xì)講解有關(guān)C#中委托和匿名委托有什么用,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

Lambda表達(dá)式從我接觸開始到現(xiàn)在變得越來越流行,Java8中開始支持、kotlin更是對C#,F(xiàn)#做了廣泛的抄襲(C#曾幾何時不也如此對待過Java嘛)。其實這都充分說明了,Lambda表達(dá)式的重要性。要搞清楚Lambda首先需要搞清楚委托。

委托:

假設(shè)現(xiàn)在我們要開發(fā)一個處理兩個整數(shù)的程序(假設(shè)先處理相加操作)

public class Worker
    {
      /// <summary>
      /// 處理兩個數(shù)
      /// </summary>
      /// <param name="a"></param>
      /// <param name="b"></param>
      /// <returns></returns>
      public int HandleTwoNumber(int a,int b)
      {
        return a + b;
      }
    }
static void Main(string[] args)
    {
      int a = int.Parse(Console.ReadLine());
      int b = int.Parse(Console.ReadLine());

      Worker worker = new Worker();
      int result = worker.HandleTwoNumber(a, b);
      Console.WriteLine(String.Format("Result:{0}", result));

      string p = Console.ReadLine();
}

如果一段時間后,我們需要它變更為減操作:

public class Worker
    {
      public int HandleTwoNumber(int a,int b)
      {
        return a - b;
      }
    }

雖然有a+b變?yōu)閍-b的變化很微小,但后續(xù)此處可能面臨多次變化(由減變?yōu)槌?........)。有變化就應(yīng)封裝變化,此處我們可以將a與b的操作行為抽象出來,用什么抽象呢?委托

public class Worker
    {
      public delegate int TwoNumberHandleMethodDelegate(int x, int y);
      public int HandleTwoNumber(int a,int b)
      {
        return a + b;
      }
    }

public delegate int TwoNumberHandleMethodDelegate(int x, int y);此處用delegate標(biāo)注,表明這是一個委托定義。如果去掉 delegate 再來觀察該定義,你會發(fā)現(xiàn)這就是一個沒有方法體的抽象方法。所以委托的含義即:與該抽象方法簽名形式相同的方法的類型。委托就是一種你定義的新數(shù)據(jù)類型,它與int、class是一樣的都是數(shù)據(jù)類型。int表示整數(shù),只要是整數(shù)都可以賦值給 int型變量;TwoNumberHandleMethodDelegate則表示,接收兩個int型參數(shù)并返回int型結(jié)果的這類方法,因此滿足上述要求的方法都可賦值給TwoNumberHandleMethodDelegate類型的變量。

如此一來Worker代碼可修改為:

public class Worker
    {
      public delegate int TwoNumberHandleMethodDelegate(int x, int y);
      public int HandleTwoNumber(int a, int b, TwoNumberHandleMethodDelegate handle)
      {
        return handle(a, b);
      }
    }

如此a、b的操作被封裝起來,所有的變化均交由調(diào)用者來處理。此處的含義:HandleTwoNumber處理a、b兩個整數(shù),具體如何處理由 handle 實施。此時你可能會問,那如何來調(diào)用該方法呢?調(diào)用如下:

private static int Add(int a, int b)
    {
      return a + b;
    }

    private static int Sub(int a, int b)
    {
      return a - b;
    }

    static void Main(string[] args)
    {
      int a = int.Parse(Console.ReadLine());
      int b = int.Parse(Console.ReadLine());
      Worker.TwoNumberHandleMethodDelegate method = new Worker.TwoNumberHandleMethodDelegate(Add);
      Worker worker = new Worker();
      int result = worker.HandleTwoNumber(10, 10,method);
       //int result = worker.HandleTwoNumber(10, 10, Sub);//簡化版
      Console.WriteLine(String.Format("Result:{0}", result));
 }

根據(jù)上面的程序可知,Main代碼塊為worker的調(diào)用者,作為調(diào)用者而言應(yīng)該最清楚自己想要讓woker做的工作。因此作為被調(diào)用者的worker而言,它只需要接收調(diào)用者M(jìn)ain給的a\b參數(shù)及執(zhí)行Main定制的算法method,然后按照算法執(zhí)行并返回結(jié)果即可。上面代碼雖然簡單,但其中的意義深遠(yuǎn),隨著編程時間的增加相信你的理解將越深刻。

委托變量在進(jìn)行賦值時除了標(biāo)準(zhǔn)的方式,還可以進(jìn)行簡化:

Worker.TwoNumberHandleMethodDelegate method = new Worker.TwoNumberHandleMethodDelegate(Add);
      Worker worker = new Worker();
      int result = worker.HandleTwoNumber(10, 10,method);
//可簡化為
// int result = worker.HandleTwoNumber(10, 10,Add);

編譯器將自動檢查Add是否符合 TwoNumberHandleMethodDelegate 的定義,如果符合允許直接將方法名賦值給委托變量。

匿名委托

通過上面的示例代碼,我們很容易發(fā)現(xiàn) TwoNumberHandleMethodDelegate method 變量被賦值為Add(Sub),因此在調(diào)用method(...)時相當(dāng)于調(diào)用Add(.....)。這樣一來就可以認(rèn)為

method與Add完全等效,既然等效那是否可以直接將Add的定義內(nèi)容賦值給method變量呢?答案是肯定的:

static void Main(string[] args)
    {

      Worker.TwoNumberHandleMethodDelegate method =private static int Add(int a, int b)
    {
      return a + b;
    };
}

但像上面這種生拉硬套是不行的,你還需要做修改。修改內(nèi)容是:因為現(xiàn)在的代碼處于Main方法中,訪問修飾符去掉,同樣static也應(yīng)去掉;同時編譯器知道你要給method賦值,那么要賦的這個值肯定滿足返回類型為int的要求,所有int在此時就多余了去掉;因為賦值之后method就等效于Add,以后調(diào)用只要通過method變量就可完成,所有Add方法名不需要去掉。如此代碼變?yōu)槿缦滦问剑?/p>

static void Main(string[] args)
    {

      Worker.TwoNumberHandleMethodDelegate method =  (int a, int b)
    {
      return a + b;
    };
}

經(jīng)過上面的修改內(nèi)容簡化了很多,但method賦值的=右端是什么東西呢?此時編譯器并不能正確識別這是一個方法,因為方法的定義需要滿足包含:訪問修身符、返回類型、方法名、參數(shù)列表、方法體五部分內(nèi)容。雖然你心里清楚這是個簡化了的方法,但是編譯器不懂你的心.........,那沒關(guān)系只要我們告訴編譯器,后面的是個簡化方法就可以了。

static void Main(string[] args)
    {

      Worker.TwoNumberHandleMethodDelegate method =  delegate(int a, int b)
    {
      return a + b;
    };
}

正如你所期望的那樣,現(xiàn)在編譯器已經(jīng)知道了=右側(cè)是你經(jīng)過簡化的方法;ok,現(xiàn)在可以正常賦值并使用了。

通過上面的定義我們發(fā)現(xiàn),用delegate標(biāo)注的簡化方法沒有一個像Add/Sub一樣固定的名字。因此我們稱這種方法叫匿名委托(我習(xí)慣稱匿名方法)。

你可能還注意到該匿名委托定義完畢后就賦值給Main代碼快中的局部變量method,因此當(dāng)超出method的作用域后,該方法就再也沒有機(jī)會調(diào)用了。這引出了匿名方法、匿名委托、匿名函數(shù)它們的最常見用法,即用來定義只需要使用一次的功能代碼。

關(guān)于“C#中委托和匿名委托有什么用”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,使各位可以學(xué)到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI