1.匿名方法和Lambda表達式中可以訪問到聲明該匿名方法或Lambda表達式所在方法中的參數或局部變數,這些變數稱為外部變數(Outer Variable),外部變數的生命周期會持續到引用匿名方法或Lambda表達式的委托實例被垃圾回收器回收為止: 2.調用匿名方法或Lambda表達式時,訪問到的 ...
1.匿名方法和Lambda表達式中可以訪問到聲明該匿名方法或Lambda表達式所在方法中的參數或局部變數,這些變數稱為外部變數(Outer Variable),外部變數的生命周期會持續到引用匿名方法或Lambda表達式的委托實例被垃圾回收器回收為止:
int myNum = 1; Action myAction = () => { Console.WriteLine(myNum); //此處的myNum即該Lambda表達式的一個外部變數 };
2.調用匿名方法或Lambda表達式時,訪問到的是外部變數最終的值,而不是聲明該匿名方法或Lambda表達式時的值;同樣,在匿名方法或Lambda表達式中修改外部變數的值後,在外部訪問到的也是修改後的值;這個特性被稱為閉包(Closure);
例如:
Action myAction = null; for (int i = 0; i < 5; i++) { myAction += () => { Console.Write(i); //由於Lambda表達式調用了該變數i,此處會將i提升為一個欄位,該欄位最後取值為5 //或使用以下方式: //int x = i; //Console.Write(x); }; } myAction(); //55555 myAction = null; for (int i = 0; i < 5; i++) { int closureIndex = i; //每次都聲明一個新的局部變數,由於Lambda表達式內調用了該新聲明的變數closureIndex,此處相當於每次都會聲明一個新的欄位,並且互不影響取值 myAction += () => { Console.Write(closureIndex); }; } myAction(); //01234 myAction = null; for (int i = 0; i < 5; i++) { int closureIndex = i; myAction +=() => { Console.Write(closureIndex); }; ++closureIndex; } myAction(); //12345
2.實際上,在聲明匿名方法和Lambda表達式時,如果在方法或表達式內部調用了外部變數,編譯器會創建一個以<>c__DisplayClass開頭的匿名嵌套類,它包含該方法或表達式內調用的所有外部變數的欄位,運行時會創建該匿名嵌套類的一個實例,並將外部變數的值傳入該實例對應的欄位,外部對該變數的所有操作也會變成對該嵌套類實例中欄位的操作;
此處以外部變數為值類型舉例,對於引用類型,匿名嵌套類中會保存該引用類型的對象:
class Program { static void Main(string[] args) { int myNum = 1; Action myAction = () => { Console.WriteLine(myNum); }; myNum = 2; myAction(); Console.Read(); } }
如果您覺得閱讀本文對您有幫助,請點一下“推薦”按鈕,您的認可是我寫作的最大動力!
作者:Minotauros
出處:https://www.cnblogs.com/minotauros/
本文版權歸作者和博客園共有,歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接,否則保留追究法律責任的權利。