使用"命名方法"實現lambda的功能(Before C#2.0)
C#2.0之前, 如果一個方法或者變數需要使用委託(delegate)時, 則之前必須建立一個命名方法, 並在需要委託的位置傳入這個名稱. 比如有一個針對int類型的資料處理, 目的是篩選出一個int數組中為奇數的元素, 或者其它一些滿足條件的元素...
public class Commom{ //命名一個委託方法 public delegate bool IntFilter(int i); //篩選出符合委託方法所要求的int, 返回一個int[] public static int[] FilterArrayOfInts(int[] ints, IntFilter filter) { ArrayList aList = new ArrayList(); foreach (int i in ints) if (filter(i)) aList.Add(i); return (int[])aList.ToArray(typeof(int)); }}
Commom類建立了一個統一的模型, 這個模型主要用於傳入一個delegate方法, 可以獲得這個delegate方法條件所要求的數字. 具體這個delegate方法可以寫在另一個類中, 可以寫很多方法, 檢測是否是奇數, 是否是偶數, 這個數是否是3的倍數...這個根據需要自己寫. 這樣寫可以提高Commom類的可重用程度.
//根據需要, 自己定義篩選方法public class MyIntFilter{ //自己定義的篩選方法1: 檢測是否是奇數 public static bool IsOdd(int i) { return ((i & 1) == 1); } //自己定義的篩選方法2: 檢測是否是偶數 public static bool IsEven(int i) { return ((i & 1) != 1); } //...根據需要還可以定義其它篩選方法}
調用MyIntFilter中的篩選方法:
int[] nums = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };//篩選出奇數int[] oddNums = Commom.FilterArrayOfInts(nums, MyIntFilter.IsOdd);//篩選出偶數int[] evenNums = Commom.FilterArrayOfInts(nums, MyIntFilter.IsEven);//測試結果, 列印出oddNums和evenNumsConsole.WriteLine("Print oddNums:");foreach (int i in oddNums) Console.Write(i + " ");Console.WriteLine("Print evenNums:");foreach (int i in evenNums) Console.Write(i + " ");使用"匿名方法"實現lambda的功能(Since C#2.0)
為什麼要使用匿名方法? 答: 偷懶. 為每一個篩選策略專門寫一個方法很多時候會令人很煩, 很多方法可能只被用到一次兩次, 且為每個方法都起一個好聽的名字並在調用這些方法的時候傳入這些方法名令人不爽, 因此C#2.0的時候可以使用"匿名方法"解決這一問題:
int[] nums = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };//篩選出奇數int[] oddNums = Commom.FilterArrayOfInts(nums, delegate(int i){ return ((i & 1) == 1);});Console.WriteLine("Print oddNums:");foreach (int i in oddNums) Console.Write(i + " ");
這時篩選出了int數組中所有的奇數, 而沒有使用MyIntFilter類裡面用於判斷是否是奇數的方法. 這種匿名的方法好處在於省去了維護程式碼中內容相關的精力.
使用lambda運算式(Since C#3.0)
lambda運算式的形式如下:
(param1, param2, param3...)=>{doSomethingWithParam1, doSomethingWithParam2, doSomethingWithParam3...}
lambda的好處在於哪裡? 最關鍵的作用可能是可讀性提高了, 簡潔明了, 只需要列出方法體中需要使用到的參數, 然後加上=>{}, 在大括弧中寫方法體. 但注意的是傳入的參數類型要滿足之前delegate定義的輸入類型, 並返回delegate定義的傳回型別.
同樣是上面的要求, 從int數組中篩選出奇數, 使用lambda運算式就是這樣寫:
int[] nums = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };//篩選出奇數int[] oddNums = Commom.FilterArrayOfInts(nums, (i) => ((i & 1) == 1));foreach (int i in oddNums) Console.Write(i + " ");何時適合使用lambda運算式
比較三種做法, 可以發現lambda運算式最簡潔. 但並不是所有的這種情況都使用lambda運算式. 對於比較複雜的演算法, 需要重用的演算法, 最好還是使用第一種"命名方法", 因為這樣做, 在任何時候任何開發人員都可以很方便的直接調用方法, 而不需考慮這些細節.
C#3.0很多新特性幾乎都暗示著是為Linq而服務的, Lambda運算式也不例外, 因為一個Linq查詢通常只需要調用一次, 不需要為每一個查詢專門命名一個方法.