天天看點

C# Lambda表達式

今天高手讨論了一個問題:

FuncList.FindAll(pNodes => pNodes.ParentID == "01")

等價于 if(pNodes.ParentID == "01") return pNodes;

實際就是一個委托函數的縮寫。

等價于 if(pNodes.ParentID == "01") return pNodes; 

總結一下:

“Lambda 表達式”是一個匿名函數,它可以包含表達式和語句,并且可用于建立委托或表達式目錄樹類型。 所有 Lambda 表達式都使用 Lambda 運算符 =>,該運算符讀為“goes to”。該 Lambda 運算符的左邊是輸入參數(如果有),右邊包含表達式或語句塊。Lambda 表達式 x => x * x 讀作“x goes to x times x”。

​​http://baike.baidu.com/view/3048187.htm​​

Windbey中為了增強對集合的通路能力, MS設計了List<T>這麼一個泛型集合, 其中有不少的增強功能,比如Foreach,ConvertAll,FindAll等等,并且為了友善使用MS在System名空間下引入了一些特制的Delegate.主要包括以下幾個:

   20     public delegate void Action<T>(T obj);   //Used by ForEach

   21     public delegate int Comparison<T>(T x, T y);  //Used by Sort

   22     public delegate TOutput Converter<TInput, TOutput>(TInput input);    //Used by ConvertAll

   23     public delegate bool Predicate<T>(T obj); //Used by FindAll

利用這些特制的Delegate,再加上匿名方法的使用,我們可以獲得更加簡潔,有效的代碼. 具體的例子我以前有過介紹.現在在Orcas中, MS加入了lambda表達式的概念. lambda表達式是匿名方法的進一步增強. 利用它可以更加友善的寫出新的方法. 而且語義上更加接近人性化.

同樣它也引入了一些特制的Delegate:

   20       public delegate T Func<T>();

   21       public delegate T Func<A0, T>(A0 arg0);

   22       public delegate T Func<A0, A1, T>(A0 arg0, A1 arg1);

   23       public delegate T Func<A0, A1, A2, T>(A0 arg0, A1 arg1, A2 arg2);

   24       public delegate T Func<A0, A1, A2, A3, T>(A0 arg0, A1 arg1, A2 arg2, A3 arg3);

和2.0中特制的Delegate對比, 你會發現它們有很多相同之處:

   20       public delegate int Comparison<T>(T x, T y);   

   21       public delegate int Func<T,T,int>(T arg0, T arg1);

   22       public delegate TOutput Converter<TInput, TOutput>(TInput input); 

   23       public delegate TOutput Func<TInput, TOutput>(TInput arg0);

   24       public delegate bool Predicate<T>(T obj);

   25       public delegate bool Func<T,bool>(T arg0);

也就是說3.0中特制的Delegate比2.0的更一般化, 2.0是3.0的特例. 是以我們完全可以将lambda表達式運用于List<T>的一些增強方法中.

Sort方法

   20  List<int> list=new List<int>();

   21  var numbers = new []{ 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };

   22  list.AddRange(numbers);

   23  list.Sort(delegate (int a, int b)

   24                {

   25                 return a.CompareTo(b);

   26                }

   27            );

   28  //use lambda

   29  list.Sort((a,b)=>a.CompareTo(b));

ConvertAll方法

   20  List<int> doubleList =list.ConvertAll<int>(delegate (int i)

   21                                             {

   22                                                 return i*2;

   23                                             });

   24  //use lambda

   25  var doubleList2=list.ConvertAll<int>(i=>i*2);

FindAll方法

   20  List<int> lowerThanFiveList =list.FindAll(delegate (int i)

   22                                                 return i<5;

   23                                             }

   24                                           );

   25  var lowerThanFiveList2=list.FindAll(i=>i<5);

從上面的例子可以看出利用lambda表達式寫出的代碼更加簡潔易懂.  (以後代碼都經過編譯測試,可不是我杜撰的.)

以上是将lambda表達式運用于2.0當中. 但是在熟悉了3.0後, 你會發現2.0中的List<T>提供的增強方法完全是多餘的了. 其實這些增強方法往往并不限于List<T>, 通常對于IEnumerable<T>對象都是适用的. 但是如果去改動IEnumable<T>接口那麼影響實在太大了,将涉及很多的類. 是以MS僅僅在List<T>中提供了這些增強方法. 不過通過List<T>的一個構造函數,你可以使得所有的IEnumerable<T>對象可以友善的轉化為List<T>,然後再利用這些方法.

這可以說是一個很取巧的方法, 不過在有了3.0的Extension Method的支援下, 就不用這麼麻煩了, 而且MS還内置了一系列更強的集合操作方法.

比如之前的FindAll方法,我們現在可以這樣寫:

   21      var numbers = new []{ 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 };

   22      var lowerThanFive=numbers.Where(i=>i<5); //never need List<T>, just operate on T[] or any other types implement IEnumerable<T>

   23      foreach (var v in lowerThanFive)

   24          Console.WriteLine(v);   

   21      var doubleList3=numbers.Select(i=>i*2);

   22      foreach (var v in doubleList3)

   23          Console.WriteLine(v); 

   21      var orderList =numbers.OrderBy(i=>i);

   22      foreach (var v in orderList)

甚至還有很多更強大的功能:

比如我要取numbers數組中最大的5個數.

   21      var big5 =numbers.OrderByDescending(i=>i).Take(5);

   22      foreach (var v in big5)

   23          Console.WriteLine(v);

通過Orcas的Extension Method和Lambda表達式, MS為集合的操作提供了更加友善強大的功能. 這裡尚未用到Standard Query Operators, 不然代碼還要被簡化. 

當然Linq現在僅僅是一個Tech Preview 版本. 尚有很多不足.尤其在智能感覺(IntelliSense)方面:

1.  var numbers = new []{ 5, 4, 1, 3, 9, 8, 6, 7, 2, 0 }; 采用這種寫法時, numbers沒有智能感覺功能.

2. 使用lambda表達式時,如果不聲明T類型,對于其中的變量沒有智能感覺功能.這點很是奇怪, Linq竟然沒有強制聲明類型.

   21     var lowerThanFive=numbers.Where<int>(i=>i<5);

   22     var lowerThanFive=numbers.Where(i=>i<5);

以上兩種寫法竟然都沒問題, 顯然在下一種寫法中變量i 無法獲得智能感覺的能力. 

3. numbers.OrderBy(...),在寫這個方法時,numbers的智能感覺中并沒有OrderBy這個方法, 但是編譯運作沒有問題.

4. lambda表達式目前不支援多條語句.