(转)C#中匿名函数、委托delegate和Action、Func、Expression、还有Lambda的关系和区别
本文转自GetPower的文章https://www.cnblogs.com/gdpw/p/9221469.html
以前一直迷迷糊糊的,现在总算搞明白。
Lambda表达式
Lamda表达式基本写法是()=>{ };
Lambda和方法一样都可以传入参数和拥有返回值。(int x)=>{return x;};
Lambda表达式并不是C#独有的,其它语言也拥有Lambda表达式。
一些文档上写Lambda是匿名函数,我认为是不对的,Lambda可以表示一个匿名函数,但是还可以来表示Linq表达式啊。
delegate委托
委托基本的使用包括声明一个委托、实例化一个委托、使用一个委托。
// 声明一个委托,类型是myDelegate, publicdelegatevoidmyDelegate( stringstr ) ; publicstaticvoidHellowChinese( stringstrChinese ) { Console.WriteLine( "Good morning,"+ strChinese); Console.ReadLine; } // 实例化 myDelegate d = newmyDelegate(HellowChinese); // 使用 d( "Mr wang");
委托很好用。当编程到一定阶段,发现只传递一个或几个int、strig类型参数是不够的,希望可以把一段代码进行传递来执行某个操作,委托提供了这样的途径,委托提供了一种可以直接访问类中方法的途径,可以将方法当作一个参数传递从而使用。
匿名函数
上面已经讲到匿名函数了。函数和方法等价,用来封装一段代码片以便随时使用,有的有返回值有的无返回值。写在类里面。
但是如果我只想把这段代码片使用一次呢?再声明一个类、一个方法实在太麻烦,这时候匿名函数就拥有了价值。用来封装一段代码、直接执行或者传递。
匿名函数总是和委托一齐使用,因为匿名函数方便了委托的使用(不用声明一个类了)
匿名方法两种实现方式:
使用Lambda表达式:
publicdelegatevoidSeep( ) ; staticvoidMain( string[] args ) {
Seep s = => { Console.WriteLine( 2); Console.WriteLine( 1); };
使用委托:
publicdelegatevoidSeep( ) ; staticvoidMain( string[] args ) {Seep see = delegate{ Console.WriteLine( 1); }; }
Action、Func、Predicate
委托需要先声明一个类型才能使用,这很麻烦,比如我在某一个命名空间声明了一个没有返回值的委托myDelegate,然后想要这个委托的时候,每个命名空间都要引用这个命名空间,这太不友好了。
微软直接就在框架中定义了三种委托:Action、Func、Predicate。这三种委托是C#的System命名空间中已经定义好的,如Action
上面已经通过匿名函数实现了不用声明类,现在通过C#预定义的委托类型实现了不用声明委托类型,那么现在就可以直接写代码片来执行了
这三种委托的相应泛型形式,可以写出如下代码:
# region无参数无返回值 // lambda实现委托实例Action a1 = => Console.WriteLine( 1); a1;// delegate实现委托实例,下面不再演示delegate。Action a2 = delegate{ Console.WriteLine( 1); }; a2;# endregion
# region有参数无返回值 Action< int> a3 = (x) => Console.WriteLine(x); a3( 1); # endregion
# region无参数有返回值的情况 Func< int> e1= => 1; varvalue1= e1; # endregion
# region有参数有返回值的情况 // 最后一个泛型是返回值的类型Func< int, int> e2 = (x) => x+ 1; intvalue2= e2( 1); // 多参数情况Func< int, int, int> e3 = (x,x2) => x+x2 + 1; intvalue3 = e3( 1, 2); # endregion
# region返回值是布尔类型 // 其实Predicate可以用Func返回值是bool来代替的。Predicate< int> p = (x) => x > 0; boolvalue4 = p( 1); # endregion
委托还有一些特性(如可加性),感兴趣的可以查阅相关资料。
Expression
Expression指的是System.Linq.Expressions,是Linq表达式,表达式树,不是Lambda表达式!!
Linq表达式用于离线集合的查询。什么是离线集合?最常用的是EF里面的DBContext,我也只用过这个,所以我不想那么多,就把Expression当成EF里面使用的就行了。
1、只能接受Lambda表达式表示的委托
2、Lambda表达式只能有一条语句。
总结
1、都是希望可以单独引用一段代码片引起的。
2、然后引用了委托delegate,delegate的使用方式为声明类型-实例化类型(传入想要使用的类的方法)-使用委托(使用方法)
3、简化操作,使用(通过Lambda或者delegate)匿名方法来直接声明一段代码片,避免声明所想使用的类的方法
4、简化操作,直接使用C#预定义的三种委托Action、Func、Predicate,连声明委托都省了。
5、本文代码:https://github.com/Beiluola/delegateDemo
注意
1、匿名函数总是和委托一起使用(三种预定义委托)
2、使用三种预定义委托就可以满足大多数需求。
3、Expression是Linq表达式树,只能使用接受一句lambda表达式。和匿名函数无关。