定义:"Lambda表达式"是一个匿名函数,是一种高效的类似于函数式编程的表达式
好处:Lambda简化了匿名委托的使用,减少开发中需要编写的代码量
。
具体内容:它可以包含表达式和语句,并且可用于创建委托或表达式目录树类型,支持带有可绑定到委托或表达式树的输入参数的内联表达式。
写法:所有Lambda表达式都使用Lambda运算符=>,该运算符读作"goes to"。Lambda运算符的左边是输入参数(如果有),右边是表达式或语句块。Lambda表达式x => x * x读作"x goes to x times x"。
接下来从例子中慢慢学习:
- namespace LambdaLearn
- {
- public class Person
- {
- public string Name { get; set; }
- public int Age { get;set; }
- }
- class Program
- {
-
- public static List<Person> PersonsList()
- {
- List<Person> persons = new List<Person>();
- for (int i = 0; i < 7; i++)
- {
- Person p = new Person() { Name = i + "人物年龄", Age = 8 - i, };
- persons.Add(p);
- }
- return persons;
- }
-
- static void Main(string[] args)
- {
- List<Person> persons0 = PersonsList();
- List<Person> persons1 = persons.Where(p => p.Age > 6).ToList();
- Person per = persons.SingleOrDefault(p => p.Age == 1);
- List<Person> persons2 = persons.Where(p => p.Name.Contains("年龄")).ToList();
- }
- }
- }
从例一可以看出一点lambda表达式的简单用法,接着往下看。
Lambda简化了匿名委托的使用,我们可以看一看下面例子怎样简化的。如果委托与事件不是很懂请看:http://blog.youkuaiyun.com/u013236878/article/details/52243017
例二:用lambda表达式简化委托
利用委托处理方法:
-
- delegate int GuangChaoshi(int a);
- static void Main(string[] args)
- {
- GuangChaoshi gwl = JieZhang;
- Console.WriteLine(gwl(10) + "");
- Console.ReadKey();
- }
-
-
- public static int JieZhang(int a)
- {
- return a + 10;
- }
利用lambda表达式处理方法:
-
- delegate int GuangChaoshi(int a);
- static void Main(string[] args)
- {
-
- GuangChaoshi gwl = p => p + 10;
- Console.WriteLine(gwl(10) + "");
- Console.ReadKey();
- }
委托跟表达式的两段代码,我们应该能明白了:其实表达式(
p => p + 10;
)中的 p 就代表委托方法中的参数,而表达式符号右边的 p+10,就是委托方法中的返回结果。
再看一个稍微复杂一点的例子:
-
- delegate int GuangChaoshi(int a,int b);
- static void Main(string[] args)
- {
- GuangChaoshi gwl = (p,z) => z-(p + 10);
- Console.WriteLine(gwl(10,100) + "");
- Console.ReadKey();
- }
- [code]csharpcode:
-
-
-
-
-
-
- delegate int GuangChaoshi(int a,int b);
- static void Main(string[] args)
- {
- GuangChaoshi gwl = (p, z) =>
- {
- int zuidixiaofei = 10;
- if (p < zuidixiaofei)
- {
- return 100;
- }
- else
- {
- return z - p - 10;
- }
-
- };
- Console.WriteLine(gwl(10,100) + "");
- Console.ReadKey();
- }
接下来看一下lambda的具体写法形式:隐式表达即没有指定参数类型(
因为编译器能够根据上下文直接推断参数的类型
)
- (x, y) => x * y
- x => x * 5
- x => { return x * 5; }
- (int x) => x * 5
- (int x) => { return x * 5; }
- () => Console.WriteLine()
看完以上内容,理解lambda表达式已经不会有太大问题了,接下来的内容稍微会深一点(至少我理解了很久Orz...)
简单了解一下lambda背景:
Lambda 用在基于方法的 LINQ 查询中,作为诸如 Where 和 Where 等标准查询运算符方法的参数。
使用基于方法的语法在 Enumerable 类中调用 Where 方法时(像在 LINQ to Objects 和 LINQ to XML 中那样),参数是委托类型 System..::.Func<(Of <(T, TResult>)>)。使用 Lambda 表达式创建委托最为方便。例如,当您在 System.Linq..::.Queryable 类中调用相同的方法时(像在 LINQ to SQL 中那样),则参数类型是 System.Linq.Expressions..::.Expression<Func>,其中 Func 是包含至多五个输入参数的任何 Func 委托。同样,Lambda 表达式只是一种用于构造表达式目录树的非常简练的方式。尽管事实上通过 Lambda 创建的对象的类型是不同的,但 Lambda 使得 Where 调用看起来类似。
背景这种想要深入研究的可以都了解一下,本文只是帮助了解lambda,这里就不多说了。
下列规则适用于 Lambda 表达式中的变量范围:
捕获的变量将不会被作为垃圾回收,直至引用变量的委托超出范围为止。
在外部方法中看不到 Lambda 表达式内引入的变量。
Lambda 表达式无法从封闭方法中直接捕获 ref 或 out 参数。
Lambda 表达式中的返回语句不会导致封闭方法返回。
Lambda 表达式不能包含其目标位于所包含匿名函数主体外部或内部的 goto 语句、break 语句或 continue 语句。
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(stringv){returnv.StartsWith("a");});
- string[]findNameB=Array.FindAll<string>(names,delegate(stringv){returnv.StartsWith("a");});
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”语句,
编译器会自动处理,否则必须加上,如下示例:
“Lambda表达式”是委托的实现方法,所以必须遵循以下规则:
1)“Lambda表达式”的参数数量必须和“委托”的参数数量相同;
2)如果“委托”的参数中包括有ref或out
修饰符,则“Lambda表达式”的参数列中也必须包括有修饰符;