C#閉包


  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    

  3.實際上,在聲明匿名方法和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/

本文版權歸作者和博客園共有,歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接,否則保留追究法律責任的權利。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM