.net 3.5 framework中的Lambda表達式

本文介紹了.net 3.5 framework中的Lambda表達式。這種表達式可以取代delegate,作爲方法指針來使用。

    C#2.0C#1.x中,需要使用delegate來定義方法指針。如下面的代碼如示:

public delegate bool Filter(int num);  // delegate類型

public int[] searchArray(int[] values, Filter filter)
{
    List
<int> result = new List<int>();
    foreach (int i in values)
    {
        
if (filter(i))
             result.Add(i);
    }            
    return result.ToArray();
}

    從上面的代碼可以看出,定義了一個叫Filter的delegate類型。然後searchArray方法的第二個參數通過Filter類型將方法指針傳到該方法中。 實際上,從IL可以看出,Filter被編譯成了一個類,而且這個類是System.MulticastDelegate的子類。但和普通類不同的是,System.MulticastDelegate及其子類只能由編譯器使用,不能直接寫在源程序中。也就是說,只能在IL中使用。

    如果要調用searchArray方法,需要先定義一個和Filter擁有同樣參數和返回值的方法,代碼如下:

public bool MyFilter(int num)
{
    
return ((num & 1== 0);
}

    下面的代碼調用了searchArray方法:

int[] intArray = new int[] { 12345678 };
int[] evenArray = searchArray(intArray, MyFilter);
foreach (int i in evenArray)
    textBox1.AppendText(i.ToString());

    C#2.0開始支持匿名方法,開發人員可以通過匿名方法用內聯代碼形式取代delegate,如上面的調用代碼可以修改爲下面的形式:

int[] intArray = new int[] { 12345678 };
int[] evenArray = searchArray(intArray, delegate(int i) { return ((i & 1== 0); });
foreach (int i in evenArray)
    textBox1.AppendText(i.ToString());

    上面的代碼將匿名方法轉換成了delegate

    然而,從C#3.0.net framework3.5)開始,支持了Lambda表達式。所謂Lambda表達式就是delegate和匿名方法的簡寫形式,Lambda表達式的語法如下:

(param1, param2 ...,paramN) =>

{

    表達式1;

    表達式2;

   return 返回值;

}

    上面語法中的param1...paramN就表示方法的參數(不用確定類型,C#編譯器會爲我們做這個工作),而{...}裏面的內容就和方法體中的內容完全一樣。

    如果delegate沒有參數,可以只寫(),如下面的方法所示:

public delegate void Method1();

public void test()
{
    Method1 method1 
= () => { int i = 4; i += 6; };
}

    如果delegate只有一個參數,參數兩邊的括號可以不寫,代碼如下:

public delegate void Method2(int i);

public void test()
{
    Method2 method2 
= i => { i++; i += 6; };
}

    如果delegate有返回值,{...}中的最後一條語句需要使用return來返回相應的值,代碼如下:


public delegate int Method3(int x, int y);

public void test()
{
    Method3 method3 
= (x, y) => { x++; y++return x + y; };
}

    如果使用Lambda表達式改寫本文開頭的例子,就會變得非常簡單,代碼如下:

int[] intArray = new int[] { 12345678 };
int[] evenArray = searchArray(intArray, i => { return (i & 1== 0; });
foreach (int i in evenArray)
    textBox1.AppendText(i.ToString());

    當然,我們也可以將Lamdba表達式賦給一個delegate,再進行傳值,代碼如下;

Filter filter = i => { return (i & 1== 0; };      
int[] intArray = new int[] { 12345678 };
int[] evenArray = searchArray(intArray, filter);
foreach (int i in evenArray)
    textBox1.AppendText(i.ToString());

    從本質上講,Lamdba表達式經過C#編譯器編譯後,仍然會變成delegate的形式,也就是說Lamdba表達式只是在語法層次上的改進,並不是IL提供的新的指令。如下面的兩行代碼是等價的:

Filter filter = i => { return (i & 1== 0; };        
Filter filter 
= delegate(int i) { return ((i & 1== 0); };

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章