C#3.0中的Lambda表达式为书写匿名方法提供了一种更加简单、更加函数化的语法。Lambda表达式的本质既是匿名方法,也即是当编译我们的程序代码时,编译器会自动帮我们将Lambda表达式转换为匿名方法。
创建Lambda表达式
Lambda表达式的书写方式是一个参数列表后跟“=>”记号,然后跟一个表达式或一个语句块,即Lambda表达式的语法格式为: (参数列表)=> 语句或语句块
Lambda表达式例子如下所示:
Code:
1. delegate int del(int
2. ...
3. return
4. int j = myDelegate(5); //j = 25
关于“参数列”,Lambda表达式的参数列可以具有显式的或隐式的类型。在一个具有显式类型的参数列表中,每个参数的类型都是显式声明的。在一个具有隐式类型的参数列表中,参数的类型是从Lambda表达式出现的上下文中推断出来的——具体来说,是当Lambda表达式被转换为一个兼容的委托类型时,该委托类型提供了参数的类型。
当Lambda表达式只有一个具有隐式类型的参数时,参数列表中的括号可以省略。即:
(param) => expr; 可以简写为: param => expr;
最后,参数列中可包含任意个参数(与委托对应),如果参数列中有0个或1个以上参数,则必须使用括号括住参数列,如下:
Code:
- "0个参数");
- "1个参数时参数列中可省略括号,值为:{0}", i);
- "包含2个参数,值为:{0}:{1}", x, y);
而“语句或语句块”中如果只有一条语句,则可以不用大括号括住,否则则必须使用大括号,如下所示:
Code:
- "只有一条语句");
- "使用大括号的表达式"); };
- //两条语句时必须要大括号
- i => { i++; Console.Write("两条语句的情况"); };
如果“语句或语句块”有返回值时,如果只有一条语句则可以不写“return”语句,编译器会自动处理,否则必须加上,如下示例:
Code:
1. class
2. {
3. delegate int AddHandler(int x, int
1.
2. static void
3. {
4. Console.Write(add(1, 3));
5. }
6. static void
7. {
8. Print((x, y) => x + y);
9. int v = x * 10; return
10. Console.Read();
11. }
12. }
Lambda表达式是委托的实现方法,所以必须遵循以下规则:
(1) Lambda表达式的参数数量必须和委托的参数数量相同;
(2)如果委托的参数中包括有ref或out修饰符,则Lambda表达式的参数列中也必须包括有修饰符;
我们来看如下例子:
Code:
1. class
2. {
3. delegate void OutHandler(out int
1.
2. static void
3. {
4. int
5. out
6. Console.Write(i);
7. }
8. static void
9. {
10. out int
11. Console.Read();
12. }
13. }
l 如果委托有返回类型,则Lambda表达式的语句或语句块中也必须返回相同类型的数据;
l 如果委托有几种数据类型格式而在Lambda表达式中编译器无法推断具体数据类型时,则必须手动明确数据类型。
由上面可见,C# 2.0规范中提到的匿名方法规范同样适用于Lambda表达式。Lambda表达式是匿名方法在功能行上的超集,提供了下列附加的功能:
l Lambda表达式允许省略参数类型并对其进行推断,而匿名方法要求参数类型必须显式地声明。
l Lambda表达式体可以是表达式或语句块,而匿名方法体只能是语句块。
l 在类型参数推导和方法重载抉择时,Lambda表达式可以被作为参数传递。
l 以一个表达式作为表达式体的Lambda表达式可以被转换为表达式树。