您好,登錄后才能下訂單哦!
本篇內容主要講解“C# Lambda Expression概念是什么”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實用性強。下面就讓小編來帶大家學習“C# Lambda Expression概念是什么”吧!
1.關于C# Lambda Expressions:
一個Lambda Expression (譯為Lambda式) 就是一個包含若干表達式和語句的匿名函數。可以被用作創建委托對象或表達式樹類型。所有的Lambda式都使用操作符“=>“,表示“goes to (轉變為)”。操作符左邊部分是輸入參數表,右邊部分是表達式或語句塊。x => x * x 讀成“x轉變為x 乘x”。Lambda式可以被賦值給一個委托類型:
C# Lambda實例1:
delegate int del(int i); del myDelegate = x => x * x; int j = myDelegate(5); //j = 25
也可以被用于創建一個表達式樹類型:
C# Lambda實例2:
using System.Linq.Expressions; //… Expression<del> = x => x *x;
操作符“=>”具有和“=”一樣的運算優先級,且為右相關(右邊先執行)。
在例1中,我們注意到委托的定義中有一個int類型的輸入參數以及int類型的返回值。例子中的Lambda式中并沒有任何類型的聲明。是編譯器為我們做了相應的隱式數據類型轉換:輸入參數類型能夠從委托的輸入參數類型隱式轉換,返回類型能夠被隱式轉換為委托的返回類型。
Lambda式不允許作為“is”和“as”操作符的左操作數出現。也就是
del myDelegate = x => x * x as string; //error
所有對于匿名方法的約束也同樣適用于Lambda式。請參閱Anonymous Methods (C# Programming Guide).
2.從表達式理解C# Lambda Expressions
由一個計算表達式組成的一個Lambda式稱之為表達式Lambda。表達式Lambda常被用于構造表達式樹。一個表達式Lambda返回計算表達式運算的結果。基本結構如下:
(input parameters) => expression //如果只有一個輸入參數時,括號可以省略。 //如果具有一個以上的輸入參數,必需加上括號。 (x) => x * x 等于 x => x * x (x, y) => x == y //可以顯式指定輸入參數的類型 (int x, string s) => s.Length > x //也可以沒有任何輸入參數 () => SomeMethod1()
上面這段代碼在Lambda式中調用了一個方法。需要注意的是,如果在創建會被其他方使用的表達式樹的時候,不宜在Lambda式中執行方法調用。比如:在SQL Server內執行。
一般來說,讓一個方法在原先設計的上下文環境以外執行沒有意義,也不能真正工作。
3.從語句理解C# Lambda Expressions
語句Lambda和表達式Lambda非常相似,只是語句被包含在大括號內:
(input parameters) => {statement;}
大括號中的語句可以是任意多條,也可以寫成多行(定義一個Lambda式也就是在定義一個匿名方法):
TestDelegate myDel = n => { string s = n + " " + "World"; Console.WriteLine(s); };
當然語句Lambda跟匿名方法一樣,無法用于創建表達式樹。
4.C# Lambda Expressions之類型猜測
當編寫一個Lambda式的時候,我們通常不需要明確指定輸入參數的類型。因為編譯器會根據Lambda體的實現,以及委托的定義來猜測類型。
舉例:如果要從一個List<int>中刪除小于100的元素
lst.RemoveAll(i => i < 100); //i會被猜測為int
通常的猜測規則如下:
◆Lambda式必須包含與委托定義中相等數量的輸入參數;
◆每個Lambda式的輸入參數必須能夠隱式轉換成委托定義中所要求的輸入參數;
◆Lambda式的返回值必須能夠隱式轉換成委托定義中的返回值。
注意:由于目前在common type system中還沒有一個“Lambda式類型”的類型。如果在有些場合提到“Lambda式的類型”,那通常表示委托的定義或者是Expression<>類型。
5.C# Lambda Expressions變量作用域
在Lambda式定義中可以引用外部變量。只要是在定義處能夠訪問到的變量,都可以在Lambda式中引用。
Lambda式的定義僅僅是定義一個匿名方法,最終會生成一個委托對象。外部變量的引用將被“捕獲”到委托對象內部,將會伴隨委托對象的整個生命周期。在委托對象生命周期結束之前該變量都不會被垃圾回收。就算外部變量已經超過了原來的作用域,也還能繼續在Lambda式中使用。所有會被引用的外部變量必須在Lambda式定義之前被顯式賦值。見下例
delegate bool D(); delegate bool D2(int i); class Test { D del; D2 del2; public void TestMethod(int input) { int j = 0; // Initialize the delegates with lambda expressions. // Note access to 2 outer variables. // del will be invoked within this method. del = () => { j = 10; return j > input; }; // del2 will be invoked after TestMethod goes out of scope. del2 = (x) => {return x == j; }; // Demonstrate value of j: // Output: j = 0 // The delegate has not been invoked yet. Console.WriteLine("j = {0}", j); // Invoke the delegate. bool boolResult = del(); // Output: j = 10 b = True //注意j在del的執行過程中被修改 Console.WriteLine("j = {0}. b = {1}", j, boolResult); } static void Main() { Test test = new Test(); test.TestMethod(5); // Prove that del2 still has a copy of // local variable j from TestMethod. //j的引用超出了原先定義的作用域 bool result = test.del2(10); // Output: True Console.WriteLine(result); Console.ReadKey(); } }
下面是關于變量作用域的規則:
◆被“捕獲”的變量在委托的生命周期結束前都不會被垃圾回收;
◆在Lambda式內部定義的變量對外不可見;
◆Lambda式無法直接捕獲一個具有ref或out描述的參數變量;
◆Lambda式中的return語句不會導致當前所在的方法返回;
◆Lambda式中不允許包含會導致跳當前執行范圍的goto,break 或 continue語句。
到此,相信大家對“C# Lambda Expression概念是什么”有了更深的了解,不妨來實際操作一番吧!這里是億速云網站,更多相關內容可以進入相關頻道進行查詢,關注我們,繼續學習!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。