委托、匿名方法、Lambda表达式的演进

假设给我们一个泛型对象List<T>,T为int类型,要求我们使用该对象方法FindAll(Predicate<T>
match)从中找出该List中的偶数,您如何实现?

  说明一下:Predicate<T>是一个泛型委托,它的原型为public delegate bool
Predicate<T>(T obj),该委托传入一个T类型对象,经逻辑判断后返回布尔值。

委托

  可能您首先想到的是用委托实现,实现方法如下:

//方法1
staticvoid
Method1()
{
//创建List<int>对象
List<int> list =new
List<int>();
list.AddRange(newint[] { 1, 2, 3, 4, 5, 6 });

//定义一个委托
Predicate<int> callback =new
Predicate<int>(IsEvenNumber);
//执行FindAll方法,该方法使用传入的委托callback来判断列表中的整数是否为偶数
List<int> evenNumbers = list.FindAll(callback);

//输出
foreach (int
i in
evenNumbers)
{

Console.Write("{0: 0}", i);
}

Console.WriteLine();
}

//判断传入的整数是否为偶数
staticbool IsEvenNumber(int i)
{
return i %2==0;
}

  这个方法不错,简单、直观,就是有些......,有些什么?代码显得有些累赘,我们仅仅是要执行“i % 2 ==
0”这样一个判断语句,却要为这一条语句专门定义一个方法和一个委托。如果该方法在别处根本未被使用,那有没有什么办法省去它呢(当然不是真正的省去,编
译后,该方法其实还存在,我们的目的是至少不用让咱们特意为这条语句输入一个方法和一个委托)?这时,我们的匿名方法该登场啦!

匿名方法

  看下面的代码:

staticvoid Method2()
{
//创建List<int>对象
List<int> list =new
List<int>();
list.AddRange(newint[] { 1, 2, 3, 4, 5, 6 });

//创建匿名方法,并将其传给FindAll方法
List<int> evenNumbers =
list.FindAll
                          (
                            delegate(int
i)
                            {
                              return
i %2==0
;
                            }

                          );

//输出
foreach (int
i in
evenNumbers)
{

Console.Write("{0: 0}", i);
}

Console.WriteLine();
}

  上例中delegate(int
i) { return
i %2==0;
}语句创建了一个内联匿名方法的委托,并传给FindAll用于判断列表中的一个整数是否为偶数。相较于Method1,我们不用特意定义一个方法和一个委托,代码是精减了不少,但看着是不是有些别扭?这个匿名方法的原型是啥样?

  么急,一点点来分析。(int
i)就是咱们定义的匿名方法的参数列表,return后返回一个布尔值,这就是该方法的返回值,这么看来,咱们的内联匿名方法的委托原型其实就是:

  delegate bool 匿名(int i)

  看起来眼熟啊!这不和Predicate<T>的原型public delegate bool
Predicate<T>(T
obj),T代入int类型后一样嘛!耶,真巧啊!是啊,不巧不行啊,您定义的匿名方法原型如果不是这个样的话,那编译器就直接报错
啦!List<int>.FindAll方法可是要求传入一个Predicate<T>的委托呢!

  嗯,这么说来,内联匿名方法的委托定义语法是这个样啦:

delegate(参数列表)
{
执行语句
}

Lambda表达式

  匿名方法的实现已经够简练了,还有啥更简练的语法来实现么?啥?还要简练?匿名方法的语法就够晕乎了,还要咋个简练法,有些BT吧!么办法,微软就是这么BT。咱们就来看看更BT的Lambda表达式吧。小二,上代码:

staticvoid Method3()
{
//创建List<int>对象
List<int> list =new
List<int>();
list.AddRange(newint[] { 1, 2, 3, 4, 5, 6 });

//创建Lambda表达式并传给FindAll方法
List<int> evenNumbers = list.FindAll

(
(int x) =>
{ return
x %2==0; }
//注意这里噢,这就是大名鼎鼎的Lambda表达式
);

//输出
foreach
(int
i in
evenNumbers)
{

Console.Write("{0: 0}", i);
}

Console.WriteLine();
}

  看到这行代码了吧:(int x) => { return x % 2 == 0; }

  这是个啥东西?Lambda表达式?对头!

  让我们来分析一下Lambda表达式的语法:别的先不管,关注一下“=>”符号。Lambda表达式被该符号分为两部分,符号的前半部分是参数列表,后半部分是执行语句,简化一下就是这样:

  参数列表=>执行语句

  眼熟不?嗯,好像、大概、似乎、肯定、因为、所以有点眼熟,对了,这结构不和匿名方法一个样吗?咱再回头看看匿名方法的语法:

  delegate(参数列表){执行语句}

  看出来了吧,Lambda表达式就是匿名方法嘛!区别仅在于用“=>”符号隔开了参数列表和执行语句而已。这么说来,语句(int x)
=> { return x % 2 == 0; }不过是为FindAll传入了一个内联匿名方法的委托,该方法的参数列表为(int
x),返回一个布尔值。

  至此,是不是值得庆祝一下?我们好像开始理解这个BT的Lambda表达式了,现在应该叫小二上二锅头了吧,我再也不想看这让人晕乎的代码了。不过,先别高兴,微软又说了,这还不是最简练的!啥?这还不简练?那你写一个更简练的看看?小二,再上一盘代码:  

staticvoid Method4()
{
//创建List<int>对象
List<int> list =new
List<int>();
list.AddRange(newint[] { 1, 2, 3, 4, 5, 6 });

//创建Lambda表达式并传给FindAll方法
List<int> evenNumbers = list.FindAll

(
x => x %2==0//看到这里,是不是有些无语了
);

//输出
foreach (int
i in
evenNumbers)
{

Console.Write("{0: 0}", i);
}

Console.WriteLine();
}

  看看这个:

  x=> x % 2 == 0

  这是我们第二次看到Lambda表达式了,只要记着Lambda表达式的语法:(参数列表=>执行语句
就行了。x就是该式的参数列表,为啥x连个参数类型都没有呢?因为编译器知道咱们要传入的方法原型是bool
Predicate<int>(int obj),自动就认为参数x的类型为int类型。同理,编译器也知道方法的返回值为布尔型,执行语句(x
% 2 ==
0)就是一个布尔型表达式,所以,编译器也认为咱们要返回的就是这个表达式了。最后Lambda表达式就像被扒光了衣服的小正太一样,裸奔在咱们的代码中
了。嗯,很好,裸一裸更健康!

  咱们的委托、匿名方法、Lambda表达式的演进之旅这样结束了。不过,好像还有一个问题,如果咱们的匿名方法不带参数,那Lambda应该怎么写?嗯,先别说答案,让我想想,是不是这样“()=>{执行语句}”?对啦,就是这样!

  刚接触这种BT的Lambda表达式,真是有些讨厌,这语法简练到“祼奔”了,但时间长了,你会发现它实在是太强大了,你一定会爱上这BT的Lambda表达式的。

时间: 2024-07-31 14:33:30

委托、匿名方法、Lambda表达式的演进的相关文章

委托-异步调用-泛型委托-匿名方法-Lambda表达式-事件

1. 委托 From: http://www.cnblogs.com/daxnet/archive/2008/11/08/1687014.html 类是对象的抽象,而委托则可以看成是函数的抽象.一个委托代表了具有相同参数列表和返回值的所有函数. [csharp] view plaincopy class Program { delegate int CalculateDelegate(int a, int b); int add(int a, int b) { return a + b; } s

18、(番外)匿名方法+lambda表达式

概念了解: 1.什么是匿名委托(匿名方法的简单介绍.为什么要用匿名方法) 2.匿名方法的[拉姆达表达式]方法定义 3.匿名方法的调用(匿名方法的参数传递.使用过程中需要注意什么) 什么是匿名方法? 匿名方法是C#2.0引入的一个新特性,它允许开发者声明自己的函数代码而无须使用委托函数. C#为委托提供一种机制,可以为委托定义匿名方法,匿名方法没有名称,编译器会定指定一个名称,匿名方法中不能使用跳转语句跳转到该匿名方法的外部,也不能跳转到该方法的内部.也不能在匿名方法外部使用的ref和out参数.

委托delegate 泛型委托action&lt;&gt; 返回值泛型委托Func&lt;&gt; 匿名方法 lambda表达式 的理解

1.使用简单委托 namespace 简单委托 { class Program { //委托方法签名 delegate void MyBookDel(int a); //定义委托 static MyBookDel myBookDel; //普通方法 public static void MathBook(int a) { Console.WriteLine("我是数学书" + a); } static void Main(string[] args) { myBookDel += Ma

C++实现委托机制(三)——lambda表达式封装

C++实现委托机制(三)——lambda表达式封装1.引言:              其实原本没打算写这一章的,不过最后想了想,嗯还是把lambda表达式也一并封装进去,让这个委托也适应lambda表达式的注册.不过在之前还是需要先了解lambda表达式. 2.lambda表达式:              如果大家还有对lambda表达式不了解的可以先去了解lambda表达式的基本语法和用法.这里我们只讲跟lambda表达式封装相关的知识.我们先来看看使用lambda表达式的好处吧:1.la

Cocos2d-x v3.0 新的事件调度方法 lambda表达式的使用

欢迎添? Cocos2d-x 交流群: 193411763 转载请注明原文出处:http://blog.csdn.net/u012945598/article/details/24603251 Cocos 2d-x 3.0 版本号中引入了C++ 11的特性.当中就包括了回调函中使用Lambda对象. 以下我们来看一段TestCpp中的代码: 在上图的触摸事件的回调函数中,共使用了三次Lambda表达式: [ ](Touch * touch,Event * event){ }; 以下我们就来介绍一

C#委托------匿名方法

namespace out_ref { //委托匿名方法 public delegate void Myadd(); class Program { static void Main(string[] args) { Say(delegate() { Console.WriteLine("哈哈,就是这么帅!"); }); Console.ReadKey(); } static void Say(Myadd myadd) { myadd(); } } } //委托匿名方法 求两个数之和

【Unity|C#】基础篇(9)——匿名函数/Lambda表达式

[学习资料] > 在线文档 官方文档:https://docs.microsoft.com/zh-cn/dotnet/csharp/ 菜鸟教程(高级教程):https://www.runoob.com/csharp/csharp-tutorial.html > 视频教程 腾讯学院.Siki学院         > 书籍 <C#图解教程>(第13章):https://www.cnblogs.com/moonache/p/7687551.html [学习内容] > 菜鸟教程

Python匿名函数——lambda表达式

如果要定义的函数很简单,一个return语句就能搞定,可以使用lambda表达式来定义, lambda表达式的语法如下: lambda parameters: expression lambda表达式并不包含return语句,凡是使用函数作为参数或返回值的地方,都可以使用lambda表达式,lambda函数的好处有: 简便,顺手 可以不给函数起名字,某些不需要重用该函数的地方,实质上不需要给函数想一个名字 下面的例1将lambda表达式定义的函数作为内置函数filter的参数, 例1. 将lam

委托匿名方法