探索C#之函數(shù)創(chuàng)建和閉包
來(lái)源:易賢網(wǎng) 閱讀:687 次 日期:2015-04-21 13:36:08
溫馨提示:易賢網(wǎng)小編為您整理了“探索C#之函數(shù)創(chuàng)建和閉包”,方便廣大網(wǎng)友查閱!

閱讀目錄:

動(dòng)態(tài)創(chuàng)建函數(shù)

匿名函數(shù)不足之處

理解c#中的閉包

閉包的優(yōu)點(diǎn)

動(dòng)態(tài)創(chuàng)建函數(shù)

大多數(shù)同學(xué),都或多或少的使用過(guò)?;仡櫹耤#中動(dòng)態(tài)創(chuàng)建函數(shù)的進(jìn)化:

C# 1.0中:

public delegate string DynamicFunction(string name);

public static DynamicFunction GetDynamicFunction()

{

return GetName;

}

static string GetName(string name)

{

return name;

}

var result = GetDynamicFunction()("mushroom");

3.0寫(xiě)慣了是不是看起來(lái)很繁瑣、落后。 剛學(xué)委托時(shí),都把委托理解成函數(shù)指針,也來(lái)看下用函數(shù)指針實(shí)現(xiàn)的:

char GetName(char p);

typedef char (*DynamicFunction)(char p);

DynamicFunction GetDynamicFunction()

{

return GetName;

}

char GetName(char p)

{

return p;

};

char result = GetDynamicFunction()('m');

對(duì)比起來(lái)和c# 1.0幾乎一模一樣了(引用/指針差別),畢竟是同一家族的。

C# 2.0中,增加匿名函數(shù):

public delegate string DynamicFunction(string name);

DynamicFunction result2 = delegate(string name)

{

return name;

};

C# 3.0中,增加Lambda表達(dá)式,華麗的轉(zhuǎn)身:

public static Func<string, string> GetDynamicFunction()

{

return name => name;

}

var result = GetDynamicFunction()("mushroom");

匿名函數(shù)不足之處

雖然增加Lambda表達(dá)式,已經(jīng)極大簡(jiǎn)化了我們的工作量。但確實(shí)有些不足之處:

var result = name => name;

這些寫(xiě)編譯時(shí)是報(bào)錯(cuò)的。因?yàn)閏#本身強(qiáng)類型語(yǔ)言的,提供var語(yǔ)法糖只是為了省去聲明確定類型的工作量。 編譯器在編譯時(shí)必須能夠完全推斷出各參數(shù)的類型才行。代碼中的name參數(shù)類

型,顯然在編譯時(shí)無(wú)法推斷出來(lái)的。

var result = (string name) => name;

Func<string, string> result2 = (string name) => name;

Expression<Func<string, string>> result3 = (string name) => name;

上面直接聲明name類型呢,很遺憾這樣也是報(bào)錯(cuò)的。代碼中已經(jīng)給出答案了,編譯器推斷不出右邊表達(dá)式是屬于Func<string, string>類型還是Expression<Func<string, string>>類型

dynamic result = name => name;

dynamic result1 = (Func<string,string>)(name => name);

用dynamic呢,同樣編譯器也分不出右邊是個(gè)委托,我們顯示轉(zhuǎn)換下就可以了。

Func<string, string> function = name => name;

DynamicFunction df = function;

這里定義個(gè)func委托,雖然參數(shù)和返回值類型都和DynamicFunction委托一樣,但編譯時(shí)還是會(huì)報(bào)錯(cuò):不能隱式轉(zhuǎn)換Func<string, string>到DynamicFunction,2個(gè)類型是不兼容的。

理解c#中的閉包

談?wù)摰絼?dòng)態(tài)創(chuàng)建函數(shù),都要牽扯到閉包。閉包這個(gè)概念資料很多了,理論部分這里就不重復(fù)了。 來(lái)看看c#代碼中閉包:

Func<Func<int>> A = () =>

{

var age = 18;

return () =>  //B函數(shù)

{

return age;

};

};

var result = A()();

上面就是閉包,可理解為就是: 跨作用域訪問(wèn)函數(shù)內(nèi)變量,也有說(shuō)帶著數(shù)據(jù)的行為。

C#變量作用域一共有三種,即:類變量,實(shí)例變量,函數(shù)內(nèi)變量。子作用域訪問(wèn)父作用域的變量(即函數(shù)內(nèi)訪問(wèn)實(shí)例/類變量)在我們看來(lái)理所當(dāng)然的,也符合我們一直的編程習(xí)慣。

例子中匿名函數(shù)B是可以訪問(wèn)上層函數(shù)A的變量age。對(duì)于編譯器而言,A函數(shù)是B函數(shù)的父作用域,所以B函數(shù)訪問(wèn)父作用域的age變量是符合規(guī)范的。

int age = 16;

void Display()

{

Console.WriteLine(age);

int age = 18;

Console.WriteLine(age);

}

上面編譯會(huì)報(bào)錯(cuò)未聲明使用,編譯器檢查到函數(shù)內(nèi)聲明age后,作用域就會(huì)覆蓋父作用域的age,(像JS就undefined了)。

Func<int> C = () =>

{

var age = 19;

return age;

};

上面聲明個(gè)同級(jí)函數(shù)C,那么A函數(shù)是無(wú)法訪C函數(shù)中的age變量的。 簡(jiǎn)單來(lái)說(shuō)就是不可跨作用域訪問(wèn)其他函數(shù)內(nèi)的變量。 那編譯器是怎么實(shí)現(xiàn)閉包機(jī)制的呢?

如上圖,答案是升級(jí)作用域,把A函數(shù)升級(jí)為一個(gè)實(shí)例類作用域。 在編譯代碼期間,編譯器檢查到B函數(shù)使用A函數(shù)內(nèi)變量時(shí),會(huì)自動(dòng)生成一個(gè)匿名類x,把原A函數(shù)內(nèi)變量age提升為x類的

字段(即實(shí)例變量),A函數(shù)提升為匿名類x的實(shí)例函數(shù)。下面是編譯器生成的代碼(精簡(jiǎn)過(guò)):

class Program1

{

static Func<Func<int>> CachedAnonymousMethodDelegate2;

static void Main(string[] args)

{

Func<Func<int>> func = new Func<Func<int>>(Program1.B);

int num = func()();

}

static Func<int> B()

{

DisplayClass cl = new DisplayClass();

cl.age = 18;

return new Func<int>(cl.A);

}

}

sealed class DisplayClass

{

public int age;

public int A()

{

return this.age;

}

}

我們?cè)賮?lái)看個(gè)復(fù)雜點(diǎn)的例子:

static Func<int, int> GetClosureFunction()

{

int val = 10;

Func<int, int> interAdd = x => x + val;

Console.WriteLine(interAdd(10));

val = 30;

Console.WriteLine(interAdd(10));

return interAdd;

}

Console.WriteLine(GetClosureFunction()(30));

輸出結(jié)果是20、40、60。 當(dāng)看到這個(gè)函數(shù)內(nèi)變量val通過(guò)閉包被傳遞的時(shí)候,我們就知道val不僅僅是個(gè)函數(shù)內(nèi)變量了。之前我們分析過(guò)編譯器怎么生成的代碼,知道val此時(shí)是一個(gè)匿名

類的實(shí)例變量,interAdd是匿名類的實(shí)例函數(shù)。所以無(wú)論val傳遞多少層,它的值始終保持著,直到離開(kāi)這個(gè)(鏈?zhǔn)?作用域。

關(guān)于閉包,在js當(dāng)中談?wù)摰谋容^多,同理,可以對(duì)比理解下:

function A() {

var age = 18;

return function () {

return age;

}

}

A()();

閉包的優(yōu)點(diǎn)

對(duì)變量的保護(hù)。想暴露一個(gè)變量值,但又怕聲明類或?qū)嵗兞繒?huì)被其他函數(shù)污染,這時(shí)就可以設(shè)計(jì)個(gè)閉包,只能通過(guò)函數(shù)調(diào)用來(lái)使用它。

邏輯連續(xù)性和變量保持。 A()是執(zhí)行一部分邏輯,A()()僅接著A()邏輯繼續(xù)走下去,在這個(gè)邏輯上下文期間,變量始終都被保持著,可以隨意使用。

更多信息請(qǐng)查看IT技術(shù)專欄

更多信息請(qǐng)查看網(wǎng)頁(yè)制作
易賢網(wǎng)手機(jī)網(wǎng)站地址:探索C#之函數(shù)創(chuàng)建和閉包
由于各方面情況的不斷調(diào)整與變化,易賢網(wǎng)提供的所有考試信息和咨詢回復(fù)僅供參考,敬請(qǐng)考生以權(quán)威部門(mén)公布的正式信息和咨詢?yōu)闇?zhǔn)!
相關(guān)閱讀網(wǎng)頁(yè)制作

2025國(guó)考·省考課程試聽(tīng)報(bào)名

  • 報(bào)班類型
  • 姓名
  • 手機(jī)號(hào)
  • 驗(yàn)證碼
關(guān)于我們 | 聯(lián)系我們 | 人才招聘 | 網(wǎng)站聲明 | 網(wǎng)站幫助 | 非正式的簡(jiǎn)要咨詢 | 簡(jiǎn)要咨詢須知 | 加入群交流 | 手機(jī)站點(diǎn) | 投訴建議
工業(yè)和信息化部備案號(hào):滇ICP備2023014141號(hào)-1 云南省教育廳備案號(hào):云教ICP備0901021 滇公網(wǎng)安備53010202001879號(hào) 人力資源服務(wù)許可證:(云)人服證字(2023)第0102001523號(hào)
云南網(wǎng)警備案專用圖標(biāo)
聯(lián)系電話:0871-65099533/13759567129 獲取招聘考試信息及咨詢關(guān)注公眾號(hào):hfpxwx
咨詢QQ:526150442(9:00—18:00)版權(quán)所有:易賢網(wǎng)
云南網(wǎng)警報(bào)警專用圖標(biāo)