认识Lambda表达式

Lambda表达式的本质是“匿名方法”,也即是当编译我们的程序代码时,“编译器”会自动帮我们将“Lambda表达式”转换为“匿名方法”,如下例:

string[] names = { "agen", "balen", "coure", "apple" };
string[] findNameA = Array.FindAll<string>(names, delegate(string v)
{
     return v.StartsWith("a");
});
string[] findNameB = Array.FindAll<string>(names, v => v.StartsWith("a"));

上面中两个FindAll方法的反编译代码如下:

string[] findNameA = Array.FindAll<string>(names, delegate (string v) {
  return v.StartsWith("a");
});
string[] findNameB = Array.FindAll<string>(names, delegate (string v) {
  return v.StartsWith("a");
});

从而可以知道“Lambda表达式”与“匿名方法”是可以划上等号的,只不过使用“Lambda表达式”输写代码看上去更直观漂亮,不是吗?


“Lambda表达式”的语法格式:

参数列 => 语句或语句块

其中“参数列”中可包含任意个参数(与委托对应),如果参数列中有0个或1个以上参数,则必须使用括号括住参数列,如下:
()=>Console.Write("0个参数")
i=>Console.Write("1个参数时参数列中可省略括号,值为:{0}",i)
(x,y) =>Console.Write("包含2个参数,值为:{0}*{1}",x,y)

而“语句或语句块”中如果只有一条语句,则可以不用大括号括住否则则必须使用,如下:

i=>Console.Write("只有一条语句")
i=>{Console.Write("使用大括号的表达式");}

//两条语句时必须要大括号
i=>{i++;Console.Write("两条语句的情况");}


如果“语句或语句块”有返回值时,如果只有一条语句则可以不输写“return”语句,编译器会自动处理,否则必须加上,如下示例:

    class Test
    {
        delegate int AddHandler(int x, int y);

        static void Print(AddHandler add)
        {
            Console.Write(add(1, 3));
        }
        static void Main()
        {
            Print((x, y) => x + y);
            Print((x, y) => { int v = x * 10; return y + v; });

            Console.Read();
        }
    }

“Lambda表达式”是委托的实现方法,所以必须遵循以下规则

1)“Lambda表达式”的参数数量必须和“委托”的参数数量相同;

2)如果“委托”的参数中包括有ref或out修饰符,则“Lambda表达式”的参数列中也必须包括有修饰符;

例子:
    class Test
    {
        delegate void OutHandler(out int x);

        static void Print(OutHandler test)
        {
            int i;
            test(out i);
            Console.Write(i);
        }
        static void Main()
        {
            Print((out int x) => x = 3);

            Console.Read();
        }
    }

注: 如果包括有修饰符,则“Lambda表达式”中的参数列中也必须加上参数的类型!

3)如果“委托”有返回类型,则“Lambda表达式”的“语句或语句块”中也必须返回相同类型的数据;

4)如果“委托”有几种数据类型格式而在“Lambda表达式”中“编译器”无法推断具体数据类型时,则必须手动明确数据类型。

例子:

(错误代码)
    class Test
    {
        delegate T AddHandler<T>(T x, T y);

        static void Print(AddHandler<int> test)
        {
            Console.WriteLine("int type:{0}", test(1, 2));
        }
        static void Print(AddHandler<double> test)
        {
            Console.WriteLine("double type:{0}", test(1d, 2d));
        }
        static void Main()
        {
            Print((x, y) => x + y);

            Console.Read();
        }
    }

当我们编译以下代码时,编译器将会显示以下错误信息:

在以下方法或属性之间的调用不明确:“ConsoleApplication1.Test.Print(ConsoleApplication1.Test.AddHandler<int>)”和“ConsoleApplication1.Test.Print(ConsoleApplication1.Test.AddHandler<double>)”


所以我们必须明确数据类型给编译器,如下:

Print((int x, int y) => x + y);

这样我们的代码就能编译通过了:)

posted @ 2008-04-18 21:53  Kingthy  阅读(7422)  评论(7编辑  收藏  举报