C#3.0中的Lambda表达式为书写匿名方法提供了一种更加简单、更加函数化的语法。Lambda表达式的本质既是匿名方法,也即是当编译我们的程序代码时,编译器会自动帮我们将Lambda表达式转换为匿名方法。
创建Lambda表达式
Lambda表达式的书写方式是一个参数列表后跟“=>”记号,然后跟一个表达式或一个语句块,即Lambda表达式的语法格式为: (参数列表)=> 语句或语句块
Lambda表达式例子如下所示:
- delegate int del(int i);
- ...
- del myDelegate = (x) => return x * x;
- int j = myDelegate(5); //j = 25
关于“参数列”,Lambda表达式的参数列可以具有显式的或隐式的类型。在一个具有显式类型的参数列表中,每个参数的类型都是显式声明的。在一个具有隐式类型的参数列表中,参数的类型是从Lambda表达式出现的上下文中推断出来的——具体来说,是当Lambda表达式被转换为一个兼容的委托类型时,该委托类型提供了参数的类型。
当Lambda表达式只有一个具有隐式类型的参数时,参数列表中的括号可以省略。即:
(param) => expr; 可以简写为: param => expr;
最后,参数列中可包含任意个参数(与委托对应),如果参数列中有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();
- }
- }
l 如果委托有返回类型,则Lambda表达式的语句或语句块中也必须返回相同类型的数据;
l 如果委托有几种数据类型格式而在Lambda表达式中编译器无法推断具体数据类型时,则必须手动明确数据类型。
由上面可见,C# 2.0规范中提到的匿名方法规范同样适用于Lambda表达式。Lambda表达式是匿名方法在功能行上的超集,提供了下列附加的功能:
l Lambda表达式允许省略参数类型并对其进行推断,而匿名方法要求参数类型必须显式地声明。
l Lambda表达式体可以是表达式或语句块,而匿名方法体只能是语句块。
l 在类型参数推导和方法重载抉择时,Lambda表达式可以被作为参数传递。
l 以一个表达式作为表达式体的Lambda表达式可以被转换为表达式树。