溫馨提示×

溫馨提示×

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

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

C#中怎么利用閉包捕獲變量

發(fā)布時間:2021-07-08 15:38:25 來源:億速云 閱讀:169 作者:Leah 欄目:編程語言

C#中怎么利用閉包捕獲變量,很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細(xì)講解,有這方面需求的人可以來學(xué)習(xí)下,希望你能有所收獲。

簡單來講,閉包允許你將一些行為封裝,將它像一個對象一樣傳來遞去,而且它依然能夠訪問到原來***次聲明時的上下文。這樣可以使控制結(jié)構(gòu)、邏輯操作等從調(diào)用細(xì)節(jié)中分離出來。訪問原來上下文的能力是閉包區(qū)別一般對象的重要特征,盡管在實(shí)現(xiàn)上只是多了一些編譯器技巧。

我們知道,在匿名方法或者lambda中,可以訪問或者修改該匿的定義范圍內(nèi)的變量。例如:

int num = 1;   Func<int> incNum = () => ++num;

其中l(wèi)ambda表達(dá)式使用了在其外部定義的變量num。我們可以認(rèn)為該段lambda語句塊構(gòu)成了一個閉包,而這個閉包捕獲了外部變量num。

好了,不說那么多讓人看著難受的定義套話了。我們進(jìn)入正題,看看在C#中變量是如何被捕獲的。來看一個例子:

public Func<String> CreateFunction()   {   String str = "我的幸運(yùn)數(shù)字是";   int num = 17;   Func<String> func = () => str + num;   return func;   }

在這個例子中,定義了一個返回一個函數(shù)的方法CreateFunction。返回的函數(shù)構(gòu)成了一個閉包,該閉包捕獲了兩個變量:String類型的str和int類型的num。

好了,我們現(xiàn)在可以這樣使用這個函數(shù)了:

Func<String>   myFunc = CreateFunction();   String result = myFunc();

我們來分析一下這兩行代碼實(shí)際都干了什么。***行很容易理解,我們把方法CreateFunction生成的匿名函數(shù)賦值給了委托myFunc。

第二行更好理解,我們執(zhí)行了myFunc,并將返回結(jié)果賦值給了變量result。我們再深入思考一下:在執(zhí)行myFunc的時候,會訪問到在CreateFunction中定義兩個變量str與num。

雖然這時CreateFunction的棧幀早就被銷毀了,其內(nèi)部定義的變量至今也“生死不明”了,但是因?yàn)槲覀冎肋@兩個變量已經(jīng)被閉包所捕獲了,所以我們堅(jiān)信這兩個變量截至目前為止還是可以訪問的!

對于str對象,鑒于它是一個引用類型,所以只要有存在某個“東西”一直保存著對它的引用,它就不會被銷毀。這樣我們完全不用擔(dān)心在我們需要它時,編譯器或運(yùn)行時會告訴我們它被弄丟了。

然而對于num,情況就有些不同了。num是一個值類型。我們知道值類型是存活在棧上的,我們也知道它所存在的那個棧幀(也就是CreateFunction的幀)在CreateFunction執(zhí)行完畢后就會被銷毀,然后其上存在的任何值類型也會被一并的銷毀,這其中當(dāng)然包括我們所關(guān)注的變量num了。

那么,我們?yōu)槭裁催€能安全的訪問num呢?C#中的變量捕獲機(jī)制究竟有什么神奇之處,可以讓值類型擁有違反常規(guī)的生存周期呢?裝箱!你可能會立刻想到,把每個值類型都裝到一個對象里,我們就可以讓這個值類型擁有和那個包裹它的對象相同的壽命了。

不過,這并不是C#實(shí)現(xiàn)者所選擇的方式!C#并不會對每個需要捕獲的值類型變量進(jìn)行裝箱操作,而是把所有捕獲的變量統(tǒng)統(tǒng)放到同一個大“箱子”里&mdash;&mdash;當(dāng)編譯器遇到需要變量捕獲的情況時,它會默默地在后臺構(gòu)造一個類型,這個類型包含了每一個閉包所捕獲的變量(包括值類型變量和引用類型變量)作為它的一個公有字段。這樣,編譯器就可以

維護(hù)那些在匿名函數(shù)或lambda表達(dá)式中出現(xiàn)的外部變量了。

更進(jìn)一步,如果我們使用ILDASM工具查看CreateFunction方法的IL代碼,我們會發(fā)現(xiàn)編譯器壓根就沒有聲明num和str變量。取而代之的是聲明了一個類型名和實(shí)例名都及其難看的包裝對象。這個玩意兒就是我們上面所說的那個被編譯器默默生成,保存了所有捕獲變量的引用的對象。

我們還可以看到,在CreateFunction方法,C#源代碼內(nèi)所有對str和num的操作,在IL中都被轉(zhuǎn)換成了對包裝對象的同名公有成員的操作。順便說一句,就連我們構(gòu)造的那個lambda表達(dá)式“() => str + num”現(xiàn)在都被編譯器轉(zhuǎn)換成了這個包裝對象的一個方法!

看完上述內(nèi)容是否對您有幫助呢?如果還想對相關(guān)知識有進(jìn)一步的了解或閱讀更多相關(guān)文章,請關(guān)注億速云行業(yè)資訊頻道,感謝您對億速云的支持。

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

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

AI