C#委托與匿名委托詳解
本來(lái)是想寫(xiě)一篇《委托與lambda表達(dá)式的前世今生》,但僅委托部分已經(jīng)寫(xiě)了很多內(nèi)容,于是就此分開(kāi)關(guān)于Lambda表達(dá)是的內(nèi)容后續(xù)再寫(xiě)吧。
不知道Lambda表達(dá)式是誰(shuí)發(fā)明的,只記得第一次接觸Lambda表達(dá)式是在使用VS2008的時(shí)候,那就先認(rèn)為是微軟發(fā)明的吧。
Lambda表達(dá)式從我接觸開(kāi)始到現(xiàn)在變得越來(lái)越流行,Java8中開(kāi)始支持、kotlin更是對(duì)C#,F(xiàn)#做了廣泛的抄襲(C#曾幾何時(shí)不也如此對(duì)待過(guò)Java嘛)。其實(shí)這都充分說(shuō)明了,Lambda表達(dá)式的重要性。要搞清楚Lambda首先需要搞清楚委托。
委托:
假設(shè)現(xiàn)在我們要開(kāi)發(fā)一個(gè)處理兩個(gè)整數(shù)的程序(假設(shè)先處理相加操作)
public class Worker
{
/// <summary>
/// 處理兩個(gè)數(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();
}
如果一段時(shí)間后,我們需要它變更為減操作:
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的操作行為抽象出來(lái),用什么抽象呢?委托
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)注,表明這是一個(gè)委托定義。如果去掉 delegate 再來(lái)觀察該定義,你會(huì)發(fā)現(xiàn)這就是一個(gè)沒(méi)有方法體的抽象方法。所以委托的含義即:與該抽象方法簽名形式相同的方法的類型。委托就是一種你定義的新數(shù)據(jù)類型,它與int、class是一樣的都是數(shù)據(jù)類型。int表示整數(shù),只要是整數(shù)都可以賦值給 int型變量;TwoNumberHandleMethodDelegate則表示,接收兩個(gè)int型參數(shù)并返回int型結(jié)果的這類方法,因此滿足上述要求的方法都可賦值給TwoNumberHandleMethodDelegate類型的變量。
如此一來(lái)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的操作被封裝起來(lái),所有的變化均交由調(diào)用者來(lái)處理。此處的含義:HandleTwoNumber處理a、b兩個(gè)整數(shù),具體如何處理由 handle 實(shí)施。此時(shí)你可能會(huì)問(wèn),那如何來(lái)調(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);//簡(jiǎn)化版
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é)果即可。上面代碼雖然簡(jiǎn)單,但其中的意義深遠(yuǎn),隨著編程時(shí)間的增加相信你的理解將越深刻。
委托變量在進(jìn)行賦值時(shí)除了標(biāo)準(zhǔn)的方式,還可以進(jìn)行簡(jiǎn)化:
Worker.TwoNumberHandleMethodDelegate method = new Worker.TwoNumberHandleMethodDelegate(Add);
Worker worker = new Worker();
int result = worker.HandleTwoNumber(10, 10,method);
//可簡(jiǎn)化為
// int result = worker.HandleTwoNumber(10, 10,Add);
編譯器將自動(dòng)檢查Add是否符合 TwoNumberHandleMethodDelegate 的定義,如果符合允許直接將方法名賦值給委托變量。
匿名委托
通過(guò)上面的示例代碼,我們很容易發(fā)現(xiàn) TwoNumberHandleMethodDelegate method 變量被賦值為Add(Sub),因此在調(diào)用method(...)時(shí)相當(dāng)于調(diào)用Add(.....)。這樣一來(lái)就可以認(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)容是:因?yàn)楝F(xiàn)在的代碼處于Main方法中,訪問(wèn)修飾符去掉,同樣static也應(yīng)去掉;同時(shí)編譯器知道你要給method賦值,那么要賦的這個(gè)值肯定滿足返回類型為int的要求,所有int在此時(shí)就多余了去掉;因?yàn)橘x值之后method就等效于Add,以后調(diào)用只要通過(guò)method變量就可完成,所有Add方法名不需要去掉。如此代碼變?yōu)槿缦滦问剑?/p>
static void Main(string[] args)
{
Worker.TwoNumberHandleMethodDelegate method = (int a, int b)
{
return a + b;
};
}
經(jīng)過(guò)上面的修改內(nèi)容簡(jiǎn)化了很多,但method賦值的=右端是什么東西呢?此時(shí)編譯器并不能正確識(shí)別這是一個(gè)方法,因?yàn)榉椒ǖ亩x需要滿足包含:訪問(wèn)修身符、返回類型、方法名、參數(shù)列表、方法體五部分內(nèi)容。雖然你心里清楚這是個(gè)簡(jiǎn)化了的方法,但是編譯器不懂你的心.........,那沒(méi)關(guān)系只要我們告訴編譯器,后面的是個(gè)簡(jiǎn)化方法就可以了。
static void Main(string[] args)
{
Worker.TwoNumberHandleMethodDelegate method = delegate(int a, int b)
{
return a + b;
};
}
正如你所期望的那樣,現(xiàn)在編譯器已經(jīng)知道了=右側(cè)是你經(jīng)過(guò)簡(jiǎn)化的方法;ok,現(xiàn)在可以正常賦值并使用了。
通過(guò)上面的定義我們發(fā)現(xiàn),用delegate標(biāo)注的簡(jiǎn)化方法沒(méi)有一個(gè)像Add/Sub一樣固定的名字。因此我們稱這種方法叫匿名委托(我習(xí)慣稱匿名方法)。
你可能還注意到該匿名委托定義完畢后就賦值給Main代碼快中的局部變量method,因此當(dāng)超出method的作用域后,該方法就再也沒(méi)有機(jī)會(huì)調(diào)用了。這引出了匿名方法、匿名委托、匿名函數(shù)它們的最常見(jiàn)用法,即用來(lái)定義只需要使用一次的功能代碼。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持我們。
上一篇:淺析C#中靜態(tài)方法和非靜態(tài)方法的區(qū)別
欄 目:C#教程
下一篇:C#實(shí)現(xiàn)動(dòng)態(tài)數(shù)據(jù)繪圖graphic的方法示例
本文標(biāo)題:C#委托與匿名委托詳解
本文地址:http://www.jygsgssxh.com/a1/C_jiaocheng/5502.html
您可能感興趣的文章
- 01-10C#通過(guò)重寫(xiě)Panel改變邊框顏色與寬度的方法
- 01-10C#實(shí)現(xiàn)實(shí)體類與字符串互相轉(zhuǎn)換的方法
- 01-10C#實(shí)現(xiàn)子窗體與父窗體通信方法實(shí)例總結(jié)
- 01-10時(shí)間戳與時(shí)間相互轉(zhuǎn)換(php .net精確到毫秒)
- 01-10基于C#實(shí)現(xiàn)簡(jiǎn)單離線注冊(cè)碼生成與驗(yàn)證
- 01-10C#開(kāi)發(fā)中的垃圾回收機(jī)制簡(jiǎn)析
- 01-10C#基于委托實(shí)現(xiàn)多線程之間操作的方法
- 01-10C#編程實(shí)現(xiàn)對(duì)象與JSON串互相轉(zhuǎn)換實(shí)例分析
- 01-10C#警惕匿名方法造成的變量共享實(shí)例分析
- 01-10C#多線程編程之使用ReaderWriterLock類實(shí)現(xiàn)多用戶讀與單用戶寫(xiě)同步


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 01-10C#通過(guò)反射獲取當(dāng)前工程中所有窗體并
- 01-10關(guān)于ASP網(wǎng)頁(yè)無(wú)法打開(kāi)的解決方案
- 01-10WinForm限制窗體不能移到屏幕外的方法
- 01-10WinForm繪制圓角的方法
- 01-10C#實(shí)現(xiàn)txt定位指定行完整實(shí)例
- 01-10WinForm實(shí)現(xiàn)仿視頻播放器左下角滾動(dòng)新
- 01-10C#停止線程的方法
- 01-10C#實(shí)現(xiàn)清空回收站的方法
- 01-10C#通過(guò)重寫(xiě)Panel改變邊框顏色與寬度的
- 01-10C#實(shí)現(xiàn)讀取注冊(cè)表監(jiān)控當(dāng)前操作系統(tǒng)已
隨機(jī)閱讀
- 04-02jquery與jsp,用jquery
- 01-10delphi制作wav文件的方法
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改


