C#进阶之全面解析Lambda表达式

引言

在实际的项目中遇到一个问题,我们经常在网上搜索复制粘贴,其中有些代码看着非常的简洁,比如Lambda表达式,但是一直没有去深入了解它的由来,以及具体的使用方法,所以在使用的时候比较模糊,其次,编程涉及面比较广,我们不可能每个方面都去精通了解,但经常运到的一些东西,必须了解其具体使用方法及使用场景,才能书写出优美、简洁、可读性强的代码。笔者通过搜索、整理资料及测试代码,详细的介绍Lambda 表达式的用法。

Lambda 表达式概念

“Lambda 表达式”(lambda expression)是一个匿名函数,可以表示为委托的代码,或者表示为表达式树的代码,它所表示的表达式树可以编译为委托。Lambda 表达式的特定委托类型取决于其参数和返回值。 不返回值的 Lambda 表达式对应于   Action   委托,具体取决于其参数数量。返回值的 Lambda 表达式对应于Func 委托,具体取决于其参数数量。

Lambda 表达式广泛用于:

将要执行的代码传递给异步方法,例如 Task.Run(Action)。

编写 LINQ 查询表达式。

创建表达式树。

C# 中委托的演变

在 C# 1.0 中,通过使用在代码中其他位置定义的方法显式初始化委托来创建委托的实例。C# 2.0 引入了匿名方法的概念,作为一种编写可在委托调用中执行的未命名内联语句块的方式。C# 3.0 引入了 Lambda 表达式,这种表达式与匿名方法的概念类似,但更具表现力并且更简练。这两个功能统称为匿名函数 。通常,面向 .NET Framework 3.5 及更高版本的应用程序应使用 lambda 表达式。

C#1.0中委托 的实现,代码如下:

delegateintCalculateHandler(intx,inty);privateintSum(intx,inty){returnx + y; }publicvoidTest(){    CalculateHandler sumHandler =newCalculateHandler(Sum);    MessageBox.Show(sumHandler(1,2).ToString());//输入结果3}

C#2.0中匿名方法 的实现,代码如下:

delegateintCalculateHandler(intx,inty);publicvoidTest(){    CalculateHandler sumHandler =delegate(intx,inty) {returnx + y; };    MessageBox.Show(sumHandler(1,2).ToString());//输入结果3}

C#3.0中Lambda 表达式 的实现,代码如下:

delegateintCalculateHandler(intx,inty);publicvoidTest(){      CalculateHandler sumHandler = (x, y) => x + y;      MessageBox.Show(sumHandler(1,2).ToString());//输入结果3}

由此可以看出微软的一步步升级,带给我们的是编程上的优美,简洁,可读性强,因此作为程序员我们要一直处于学习的路上。

Lambda 表达式使用

C#的Lambda 表达式都使用 Lambda 运算符 =>,该运算符读为“goes to”,若要创建 Lambda 表达式,需要在 lambda 运算符左侧指定输入参数(如果有),然后在另一侧输入表达式或语句块。例如,单行 Lambda 表达式x => x * x 指定名为x 的参数并返回x 的平方值。

在介绍Lambda 表达式使用之前我们先了解.Net为我们定义好的Action<T>和Func<T>两个泛型委托。

Action<T>泛型委托

Action<T>委托表示引用一个返回类型为Void的方法。这个委托存在不同的变体,可以传递之多16个不同的参数类型。同时,没有泛型参数的Action类可以调用没有参数的方法。例如,Action<in T>表示有一个输入参数的方法,Action<in T1,in T2>表示有两个输入参数的方法。

Func<T>泛型委托

Func<T>可以以类似的方法使用。不过Func<T>允许调用带返回参数的方法。Func<T>也有不同的变体,之多可以传递16个参数和一个返回类型。例如:Func<out TResult>委托类型可以无参的带返回类型的方法,Func<in T1,inT2,out Tresult>表示带两个参数和一个返回类型的方法。

Func可以表示带输出的方法,T可以有多个,且只有最后一个表示输出 即最后一个是返回类型。Func中的字符in、out在实际代码中是不会出现的。

表达式 Lambda

表达式位于=> 运算符右侧的 Lambda 表达式称为“表达式 lambda”。具体形式:(input-parameters) => expression ,表达式 lambda 会返回表达式的结果。

具体事例,代码如下:

public Action SuccessPrompt =() => MessageBox.Show("执行成功!","提示", MessageBoxButtons.OK, MessageBoxIcon.Information);//没有参数,括号不能省略public Func Compare = (x, y) => x > y;//判断x是否大于y

语句 Lambda

语句 Lambda 与表达式 lambda 表达式类似,只是语句括在大括号中,具体形式:(input-parameters) => { statement; } 。语句 lambda 的主体可以包含任意数量的语句;但是,实际上通常不会多于两个或三个。

具体事例代码如下:

publicAction Prompt = prompt =>        {            MessageBox.Show(prompt,"提示", MessageBoxButtons.OK, MessageBoxIcon.Information);        };//仅当 Lambda 只有一个输入参数时,括号才是可选的;否则括号是必需的

异步 Lambda

通过使用 async 和 await 关键字,你可以轻松创建包含异步处理的 lambda 表达式和语句。其中async 和await 关键字是在 C# 5 中引入的。

await关键字

await 运算符应用于异步方法中的任务,在方法的执行中插入挂起点,直到所等待的任务完成。任务表示正在进行的工作。

await 仅可用于由 async 关键字修改的异步方法中。使用async 修饰符定义并且通常包含一个或多个await 表达式的这类方法称为异步方法。

async修饰符

使用async 修饰符可将方法、lambda 表达式或匿名方法指定为异步。如果对方法或表达式使用此修饰符,则其称为异步方法。

使用异步 lambda 添加事件处理程序。若要添加此处理程序,请在lambda 参数列表前添加async 修饰符,代码如下:

publicpartialclassForm1:Form{publicForm1(){        InitializeComponent();        button1.Click +=async(sender, e) =>        {awaitExampleMethodAsync();            textBox1.Text +="\r\nControl returned to Click event handler.\n";        };    }privateasyncTaskExampleMethodAsync(){// The following line simulates a task-returning asynchronous process.awaitTask.Delay(1000);//Task.Delay方法只会延缓异步方法中后续部分执行时间,当程序执行到await表达时,一方面会立即返回调用方法,执行调用方法中的剩余部分,这一部分程序的执行不会延长。另一方面根据Delay()方法中的参数,延时对异步方法中后续部分的执行。}}

Lambda 表达式和元组

自 C# 7.0(对应.NET Framework4.7 和Visual Studio 2017 )起,C# 语言提供对元组 的内置支持。可以提供一个元组作为 Lambda 表达式的参数,同时 Lambda 表达式也可以返回元组。在某些情况下,C# 编译器使用类型推理来确定元组组件的类型。 可通过用括号括住用逗号分隔的组件列表来定义元组,通常,元组字段命名为Item1 、Item2 等等。但是,可以使用命名组件定义元组。

事例代码如下:

publicvoidTest1(){    Func<(int,int,int), (int,int,int)> doubleItem = ns => (2* ns.Item1,2* ns.Item2,2* ns.Item3);varitemList = (1,2,3);varresultDItemList = (itemList);//结果为[1, 2, 9]}publicvoidTest2(){    Func<(intx,inty,intz), (int,int,int)> doubleItem = ns => (2* ns.x,2* ns.y,2* ns.z);varitemList = (1,2,3);varresultDItemList = (itemList);//结果为[1, 2, 9]}

含标准查询运算符的 Lambda

在其他实现中,LINQ to Objects 有一个输入参数,其类型是泛型委托 Func 系列中的一种。这些委托使用类型参数来定义输入参数的数量和类型,以及委托的返回类型。Func 委托对于封装用户定义的表达式非常有用,这些表达式将应用于一组源数据中的每个元素。

标准查询运算符是组成 LINQ 模式的方法。这些方法中的大多数都作用于序列;其中序列指其类型实现IEnumerable<T> 接口或IQueryable<T> 接口的对象。标准查询运算符提供包括筛选、投影、聚合、排序等在内的查询功能。

共有两组 LINQ 标准查询运算符,一组作用于类型IEnumerable<T> 的对象,另一组作用于类型IQueryable<T> 的对象。构成每个集合的方法分别是Enumerable 和Queryable 类的静态成员。这些方法被定义为作为方法运行目标的类型的扩展方法 。这意味着可以使用静态方法语法或实例方法语法来调用它们。

具体事例代码如下:

publicclassSutdent{            public string Id {get;set; }            public string Name {get;set; }            publicintAge {get;set; }        }staticvoidMain(string[] args)        {List studentList =newList();            studentList.Add(newSutdent {Id ="001", Name ="张三", Age =18});            studentList.Add(newSutdent {Id ="002", Name ="李四", Age =19});            studentList.Add(newSutdent {Id ="003", Name ="王五", Age =16});            studentList.Add(newSutdent {Id ="004", Name ="赵六", Age =17});List list1 = studentList.FindAll(st => st.Age >17);//选择年龄大于17的所有学生List list2 = studentList.Where(st => st.Age >17).ToList();//选择年龄大于17的所有学生studentList.Sort((st1,st2)=>st2.Age-st1.Age);//按Age降序排列List list3 = studentList.Select(st => st.Name).ToList();//选择列表中的所有名字}

Lambda 表达式中的类型推理

编写 Lambda 时,通常不必为输入参数指定类型,因为编译器可以根据 Lambda 主体、参数类型以及 C# 语言规范中描述的其他因素来推断类型。

lambda 类型推理的一般规则如下:

Lambda 包含的参数数量必须与委托类型包含的参数数量相同。

Lambda 中的每个输入参数必须都能够隐式转换为其对应的委托参数。

Lambda 的返回值(如果有)必须能够隐式转换为委托的返回类型。

总结

通过上边的讲解,我们可以看出Lambda表达式的用法非常的简单,特别在 标准查询运算符中应用非常广泛,提高了编程效率,且写出的代码非常的简洁。文中若有不足之处,还望海涵,博文写作不易希望多多支持,后续会更新更多内容,感兴趣的朋友可以加关注,欢迎留言交流!

看我主页简介免费C++学习资源,视频教程、职业规划、面试详解、学习路线、开发工具

每晚8点直播讲解C++编程技术。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容