天天看点

C#委托与Lambda表达式

本文介绍了.net 3.5 framework中的Lambda表达式。这种表达式可以取代delegate,作为方法指针来使用。

    在C#2.0及C#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[] { 1, 2, 3, 4, 5, 6, 7, 8 };

int[] evenArray = searchArray(intArray, MyFilter);

foreach (int i in evenArray)

textBox1.AppendText(i.ToString());

    从C#2.0开始支持匿名方法,开发人员可以通过匿名方法用内联代码形式取代delegate,如上面的调用代码可以修改为下面的形式:

int[] evenArray = searchArray(intArray, delegate(int i) { return ((i & 1) == 0); });

    上面的代码将匿名方法转换成了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);

    Method2 method2 = i => { i++; i += 6; };

    如果delegate有返回值,{...}中的最后一条语句需要使用return来返回相应的值,代码如下:

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

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

如果使用Lambda表达式改写本文开头的例子,就会变得非常简单,代码如下:

int[] evenArray = searchArray(intArray, i => { return (i & 1) == 0; });

当然,我们也可以将Lamdba表达式赋给一个delegate,再进行传值,代码如下;

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

int[] evenArray = searchArray(intArray, filter);

    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); };