成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

C# 本地函數與 Lambda 表達式

開發 后端
C# 局部函數通常被視為 lambda 表達式的進一步增強。雖然功能是相關的,但也存在重大差異。

[[420552]]

本文轉載自微信公眾號「DotNET技術圈」,作者Vladimir Sadov。轉載本文請聯系DotNET技術圈公眾號。

C# 局部函數通常被視為 lambda 表達式的進一步增強。雖然功能是相關的,但也存在重大差異。

Local Functions 是嵌套函數[1]功能的 C# 實現。一種語言在支持 lambdas 之后獲得對嵌套函數的支持幾個版本是有點不尋常的。通常情況相反。

Lambda 或一般的一流函數需要實現未在堆棧上分配且生命周期與需要它們的功能對象相關聯的局部變量。如果不依賴垃圾收集或通過捕獲列表等解決方案將變量所有權的負擔減輕給用戶,則幾乎不可能正確有效地實現它們。對于某些早期語言來說,這是一個嚴重的阻塞問題。嵌套函數的簡單實現不會遇到這種復雜情況,因此一種語言更常見的是僅支持嵌套函數而不支持 lambda。

無論如何,由于 C# 長期以來一直使用 lambda,因此從差異和相似之處來看本地函數確實是有意義的。

Lambda 表達式

Lambda 表達式x => x + x是抽象地表示一段代碼以及它如何綁定到其詞法環境中的參數和變量的表達式。作為代碼的抽象表示,lambda 表達式不能單獨使用。為了使用由 lambda 表達式生成的值,需要將其轉換為更多內容,例如委托或表達式樹。

  1. using System; 
  2. using System.Linq.Expressions; 
  3.  
  4. class Program 
  5.     static void Main(string[] args) 
  6.     { 
  7.         // can't do much with the lambda expression directly 
  8.         // (x => x + x).ToString();  // error 
  9.  
  10.         // can assign to a variable of delegate type and invoke 
  11.         Func<intint> f = (x => x + x); 
  12.         System.Console.WriteLine(f(21)); // prints "42" 
  13.  
  14.         // can assign to a variable of expression type and introspect 
  15.         Expression<Func<intint>> e = (x => x + x); 
  16.         System.Console.WriteLine(e);     // prints "x => (x + x)" 
  17.     } 

有幾點值得注意:

  • lambdas 是產生函數值的表達式。
  • lambda 值的生命周期是無限的——從 lambda 表達式的執行開始,只要存在對該值的任何引用。這意味著 lambda 從封閉方法中使用或“捕獲”的任何局部變量都必須在堆上分配。由于 lambda 值的生命周期不受產生它的堆棧幀的生命周期的限制,因此不能在該堆棧幀上分配變量。
  • lambda 表達式要求在執行 lambda 表達式時明確分配主體中使用的所有外部變量。lambda 的第一次和最后一次使用的時刻很少是確定性的,因此該語言假設 lambda 值可以在創建后立即使用,只要它們是可訪問的。因此,一個 lambda 值在創建時必須是完全可用的,并且它使用的所有外部變量都必須明確分配。
  1. int x; 
  2.  
  3.      // ERROR: 'x' is not definitely assigned 
  4.      Func<int> f = () => x; 
  • lambdas 沒有名字,也不能被象征性地引用。特別是 lambda 表達式不能遞歸聲明。

注意:可以通過調用分配給 lambda 的變量或傳遞給自應用其參數的高階方法來創建遞歸 lambda(請參閱:C# 中的匿名遞歸[2]),但這不會表達真正的自我參照。

本地函數

局部函數基本上只是在另一個方法中聲明的方法,作為一種降低方法對其聲明范圍內的可見性的方法。

自然地,局部函數中的代碼可以訪問其包含范圍內可訪問的所有內容——局部變量、封閉方法的參數、類型參數、局部函數。一個值得注意的例外是外部方法標簽的可見性。封閉方法的標簽在局部函數中不可見。這只是普通的詞法范圍,它的工作原理與 lambdas 相同。

  1. public class C 
  2.     object o; 
  3.  
  4.     public void M1(int p) 
  5.     { 
  6.         int l = 123; 
  7.  
  8.         // lambda has access to o, p, l, 
  9.         Action a = ()=> o = (p + l); 
  10.     } 
  11.  
  12.     public void M2(int p) 
  13.     { 
  14.         int l = 123; 
  15.  
  16.         // Local Function has access to o, p, l, 
  17.         void a() 
  18.         { 
  19.           o = (p + l); 
  20.         } 
  21.     } 

與 lambda 的明顯區別在于局部函數具有名稱并且可以在沒有任何間接方式的情況下使用。局部函數可以是遞歸的。

  1. static int Fac(int arg) 
  2.     int FacRecursive(int a) 
  3.     { 
  4.         return a <= 1 ? 
  5.                     1 : 
  6.                     a * FacRecursive(a - 1); 
  7.     } 
  8.  
  9.     return FacRecursive(arg); 

與 lambda 表達式的主要語義區別在于局部函數不是表達式,它們是聲明語句。在代碼執行方面,聲明是非常被動的實體。事實上,聲明并沒有真正被“執行”。與標簽等其他聲明類似,局部函數聲明只是將函數引入包含范圍,而無需運行任何代碼。

更重要的是,無論是聲明本身還是嵌套函數的常規調用都不會導致對環境的不確定捕獲。在簡單和常見的情況下,如普通的調用/返回場景,捕獲的局部變量不需要進行堆分配。

例子:

  1. public class C 
  2. {     
  3.     public void M() 
  4.     { 
  5.         int num = 123; 
  6.  
  7.         // has access to num 
  8.         void  Nested() 
  9.         { 
  10.            num++; 
  11.         } 
  12.  
  13.         Nested(); 
  14.  
  15.         System.Console.WriteLine(num); 
  16.     } 

上面的代碼大致相當于(反編譯):

  1. public class C 
  2.   // A struct to hold "num" variable. 
  3.   // We are not storing it on the heap, 
  4.   // so it does not need to be a class 
  5.   private struct <>c__DisplayClass0_0 
  6.   { 
  7.       public int num; 
  8.   } 
  9.  
  10.   public void M() 
  11.   { 
  12.       // reserve storage for "num" in a display struct on the _stack_ 
  13.       C.<>c__DisplayClass0_0 env = default(C.<>c__DisplayClass0_0); 
  14.  
  15.       // num = 123 
  16.       env.num = 123; 
  17.  
  18.       // Nested() 
  19.       // note - passes env as an extra parameter 
  20.       C.<M>g__a0_0(ref env); 
  21.  
  22.       // System.Console.WriteLine(num) 
  23.       Console.WriteLine(env.num); 
  24.   } 
  25.  
  26.     // implementation of the the "Nested()"
  27.     // note - takes env as an extra parameter 
  28.     // env is passed by reference so it's instance is shared 
  29.     // with the caller "M()" 
  30.     internal static void <M>g__a0_0(ref C.<>c__DisplayClass0_0 env) 
  31.     { 
  32.         env.num += 1; 
  33.     } 

請注意,上面的代碼直接調用了“Nested()”的實現(不是通過委托間接),并且沒有在堆上引入顯示存儲的分配(就像 lambda 會那樣)。局部變量存儲在結構中而不是類中。的生命周期num并沒有因為它在 中的使用而改變Nested(),所以它仍然可以在棧上分配。M()可以只通過num引用傳遞,但編譯器使用結構體進行打包,因此它可以傳遞所有本地變量,就像num只使用一個 env 參數一樣。

另一個有趣的一點是,只要本地函數在給定范圍內可見,就可以使用它們。這是一個重要的事實,使遞歸和相互遞歸的場景成為可能。這也使得本地函數聲明在源代碼中的確切位置在很大程度上變得不重要。

例如,封閉方法的所有變量必須在調用讀取它們的本地函數時明確分配,而不是在其聲明時。實際上,如果調用可以更早發生,那么在聲明時提出該要求將沒有任何好處。

  1. public void M() 
  2.     // error here - 
  3.     // Use of unassigned local variable 'num' 
  4.     Nested(); 
  5.  
  6.     int num; 
  7.  
  8.     // whether 'num' is assigned here or not is irrelevant 
  9.     void  Nested() 
  10.     { 
  11.        num++; 
  12.     } 
  13.  
  14.     num = 123; 
  15.  
  16.     // no error here - 'num' is assigned 
  17.     Nested(); 
  18.  
  19.     System.Console.WriteLine(num); 

此外 - 如果從未使用過局部函數,它也不會比一段無法訪問的代碼和任何變量更好,否則它會使用,不需要分配。

  1. public void M() 
  2. {         
  3.     int num; 
  4.  
  5.     // warning - Nested() is never used. 
  6.     void  Nested() 
  7.     { 
  8.        // no errors on unassigned 'num'
  9.        // this code never runs. 
  10.        num++; 
  11.     } 

那么,局部函數的目的是什么?

與 lambdas 相比,局部函數的主要價值主張是局部函數在概念上和運行時開銷方面都更簡單。

Lambda 可以很好地充當一類函數[3]的角色,但有時您只需要一個簡單的助手。分配給局部變量的 Lambda 可以完成這項工作,但存在間接開銷、委托分配和可能的閉包開銷。私有方法也有效,調用成本更低,但存在封裝問題,或缺乏封裝。這樣的助手對包含類型中的每個人都是可見的。太多這樣的幫手會導致嚴重的混亂。

局部函數非常適合這種情況。調用本地函數的開銷與調用私有方法的開銷相當,但使用其他不應調用的方法污染包含類型沒有問題。

http://mustoverride.com/local_functions/

References

[1] 嵌套函數: https://en.wikipedia.org/wiki/Nested_function

[2] C# 中的匿名遞歸: https://blogs.msdn.microsoft.com/wesdyer/2007/02/02/anonymous-recursion-in-c/ 

[3] 一類函數: https://en.wikipedia.org/wiki/First-class_function

 

責任編輯:武曉燕 來源: DotNET技術圈
相關推薦

2009-08-27 09:44:59

C# Lambda表達

2024-03-25 13:46:12

C#Lambda編程

2009-08-27 09:57:50

C# Lambda表達

2009-08-26 16:17:23

C# Lambda表達

2009-07-09 09:51:07

Lambda表達式C#

2011-05-20 17:50:45

C#

2020-10-16 06:40:25

C++匿名函數

2010-09-14 14:05:42

C#委托

2009-09-14 13:57:20

C# Lambda表達Lambda表達式

2022-11-07 07:11:19

C#lambda函數

2010-10-19 10:03:02

Lambda表達式

2009-08-31 17:11:37

Lambda表達式

2009-08-07 15:41:39

C#正規表達式

2023-11-02 08:25:58

C++Lambda

2009-09-09 13:01:33

LINQ Lambda

2009-09-15 15:18:00

Linq Lambda

2022-12-05 09:31:51

接口lambda表達式

2009-07-01 09:56:10

C#3.0

2009-09-11 09:48:27

Linq Lambda

2009-08-17 13:56:28

C#正則表達式入門
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 在线视频h| 欧美日韩一 | 久久久精品一区 | 99精品欧美 | 欧美一区二区三区在线 | 第一av | 美女逼网站 | 搞av.com| 播放一级毛片 | 久久不卡 | 亚洲成人免费电影 | 久久久久久综合 | 国产伦精品一区二区三区四区视频 | 激情久久av一区av二区av三区 | 91新视频 | 综合久久网 | 中文字幕成人av | 免费成人高清在线视频 | 国产精品18毛片一区二区 | 最近免费日本视频在线 | 天堂一区二区三区四区 | 91午夜在线 | h片免费看 | 九九热在线精品视频 | 99在线资源 | 农村妇女毛片精品久久久 | 午夜不卡福利视频 | 日韩欧美不卡 | 一级黄色录像片子 | 国产精品久久久久久久久免费丝袜 | av片在线免费看 | 亚洲精品一二区 | 日韩精品激情 | 欧美大片一区 | 国产精品国色综合久久 | 亚洲视频免费在线观看 | 日韩成人高清 | 精品国产1区2区3区 一区二区手机在线 | 亚洲精品v日韩精品 | 中文字幕视频在线 | 九九精品在线 |