天天看點

C# Lambda表達式

自C#3.0開始,就可以使用一種新文法把實作代碼賦予委托:Lambda表達式。隻要有委托參數類型的地方,就可以使用Lambda表達式。

Lambda表達式的文法比匿名方法簡單。如果所調用的方法有參數,且不需要參數,匿名方法的文法就比較簡單,因為這樣不需要提供參數。

直接上一些例子,讀者可以自行運作調試。

static void Main(string[] args)
{
    //1、lambda之hello world.
    Func<string, string> lambda = param =>
    {
        return param;
    };
                                                                          
    Console.WriteLine(lambda("hello lambda."));
                                                                          
    //2、lambda之參數
    Func<double, double, double, double> cube = (double x, double y, double z) => x * y * z;
    Console.WriteLine(cube(1, 2, 3));//輸出6
                                                                          
    //更進一步,參數的類型可以省略
    Func<double, double, double, double> cude1 = (x, y, z) => x * y * z;
    Console.WriteLine(cude1(2, 4, 6));//輸出48
                                                                          
    //3、lambda之調用外部參數
    int someValue = 4;
    Func<double, double, double, double> f = (x, y, z) => x + y + z + someValue;
    Console.WriteLine(f(1, 2, 3));//1+2+3+4=10
}      

在winform程式設計中,我們經常給某個按鈕添加Click事件,最原始的寫法是直接輕按兩下按鈕,生成Click事件的代碼,我們就可以直接在裡面寫方法了。其實這是vs替我們做了很多工作,其中就有在designer.cs檔案中,生成了 this.button1.Click += new System.EventHandler(this.button1_Click);這樣的代碼,事件要注冊才能觸發。

我們采用匿名委托的寫法,在窗體的Form_Load事件中添加如下代碼,則可以觸發button1的Click事件。

this.button1.Click += delegate(object ss, EventArgs ee)
{
    MessageBox.Show("hello,this is button1.Click event...");
};      

再看看Lambda表達式的用法。

this.button1.Click += (ss, ee) =>
{
    MessageBox.Show("hello,this is button1.Click event...");
};      

是不是更加簡潔?

我們再看一下,從匿名方法到Lambda簡寫的演化過程,借一張圖。

C# Lambda表達式

從這個演化過程,可以看出C#語言的發展變化過程。

下面詳細示範一下Lambda的應用執行個體。

首先第一個執行個體我們應用Lambda對一個List<Student>清單進行循環、查找、排序等操作。

首先我們提供一個實體類:

public class Student
{
    public Student(string name, int age)
    {
        this.Name = name;
        this.Age = age;
    }
                                              
    public string Name { get; set; }
    public int Age { get; set; }
}      

然後我們初始化一個List,

//初始化3個Student對象
var students = new List<Student>()
{
    new Student("zhang3",18),
    new Student("li4",22),
    new Student("wang5",20)
};      
//定義1個列印函數
Action<Student> print = student => Console.WriteLine(string.Concat(student.Name, ":", student.Age));      

那麼,我們就可以先循環輸出一下:

students.ForEach(print);//循環列印輸出      

如果我們要找到年齡大于20歲的學生,可以這樣:

students.FindAll(student => student.Age > 20).ForEach(print);      

進一步,我們按照學生的年齡進行排序:

students.Sort((f1, f2) => f1.Age.CompareTo(f2.Age));
students.ForEach(print);//年齡從小到大排序後輸出      

現在,我們按照在students清單裡再加一個學生:

var students = new List<Student>()
{
    new Student("zhang3",18),
    new Student("li4",22),
    new Student("wang5",20),
    new Student("zhao6",20)
};      

現在要按照年齡進行分組,并統計出各個年齡階段的人數:

var result = students.GroupBy(x => x.Age).Select(x => string.Concat(x.Key, ":", x.Count()));//先進行分組,再進行投影
result.ToList().ForEach(x => Console.WriteLine(x.ToString()));//循環輸出得到結果      

再提供一個綜合性的例子,這是論壇裡的caozhy版主提供的一個例子。用Lambda表達式實作快速排序。

Func<Func<int, int, bool>, Func<int[], int[]>> filter = x => new Func<int[], int[]>(y => y.Skip(1).Where(z => x(y[0], z)).ToArray());
Func<int[], int[]> qsort = x => x;
Func<int[], int[]> lesser = dt => filter((x, y) => y < x)(dt);
Func<int[], int[]> greater = dt => filter((x, y) => y >= x)(dt);
       
qsort = dt => dt.Length > 1
    ? qsort(lesser(dt)).Concat(new int[] { dt[0] }).Concat(qsort(greater(dt))).ToArray() : dt;
       
int[] data = { 4, 3, 1, 4, 6, 7, 5, 9, 3, 11, 1, 2, 11 };
var result = qsort(data);
result.ToList().ForEach(x => Console.WriteLine(x.ToString()));//排完序後輸出