C# 知识回顾 - Lambda
序

它是第十一个希腊字母,一个拥有失意、无奈、孤独、低调等含义的流行符号,也指示一款称为“半条命”的游戏。
不过,这次我所讲的是 C# 中的 Lambda。
目录
- Lambda 简介
- Lambda 表达式
- Lambda 语句
- 异步 Lambda
- 在 LINQ 中使用 Lambda
- Lambda 中的类型推断
- Lambda 中的变量使用范围
- Lambda 的特点
Lambda 简介
Lambda 表达式,是一种简化的匿名函数,可用于创建委托或表达式目录树。其次,你也可以将 Lambda 表达式作为参数进行传递,或者将它作用于函数调用值调用后返回的一个函数来使用。我们经常在 LINQ 中使用 Lambda 表达式。
创建 Lambda 表达式的简单语法形式:输入参数 => 表达式或语句块。其中,=> 为 Lambda 运算符,可读作“goes to” 。

delegate int MyDel(int x);
static void Main(string[] args)
{
MyDel myDel = x => x++;
var j = myDel(5);
}

创建表达式树:
Expression<MyDel> myDel = x => x++;
=> 运算符和 = 运算符 (赋值运算符),具有相同的优先级,并且都是右结合运算。
我们经常在 LINQ 查询中使用 Lambda 表达式,如作为 Where<TSource> 的参数。该方法有多个重载,这里只列举了其中一个。

1 //
2 // 摘要:
3 // 基于谓词筛选值序列。
4 //
5 // 参数:
6 // source:
7 // 要筛选的 System.Collections.Generic.IEnumerable<T>。
8 //
9 // predicate:
10 // 用于测试每个元素是否满足条件的函数。
11 //
12 // 类型参数:
13 // TSource:
14 // source 中的元素的类型。
15 //
16 // 返回结果:
17 // 一个 System.Collections.Generic.IEnumerable<T>,包含输入序列中满足条件的元素。
18 //
19 // 异常:
20 // System.ArgumentNullException:
21 // source 或 predicate 为 null。
22 public static IEnumerable<TSource> Where<TSource>(this IEnumerable<TSource> source, Func<TSource, bool> predicate);

参数是委托类型 Func<TSource, bool> predicate),这里使用 Lambda 表达式进行创建我想应该是最合适的。还有,假如参数类型为抽象类的 System.Linq.Expressions.Expression<Func>,其中 Func 委托是重载具有十六个参数的,你也可以使用 Lambda 表达式创建对应的表达式树。
【注意】在 is 或 as 运算符的左侧不允许使用 Lambda 表达式。
Lambda 表达式
表达式在 => 运算符右侧,称“lambda 表达式”。lambda 表达式常用于 LINQ 和构建表达式树,它也允许返回结果。
基本形式:( 输入参数 ) => 表达式 。
如:
( ) => true;
x => x == 1;
(x) => x == 1;
(x, y) => x == y;
【备注】当 lambda 表达式有且只有一个输入参数的时侯,括号(“()”)才是可选的。 括号内存在多个输入参数时使用“,”进行分割。
你也可以选择显式指定类型,一般只有在编译器难以或无法准确推断输入类型的时候。
Func<int, int, bool> func = (int x, int y) => x == y;
这里使用空括号(“()”)指定零个输入参数,并且可以在 Lambda 的主体包含一个或多个方法进行调用。
() => YourMethod()
Lambda 语句
基本形式:( 输入参数 ) => { 表达式 } 。
lambda 语句的主体可以由任意数量的普通语句组成,不过,我们一般写的语句不多(三个左右吧)。

delegate void MyDel(string s);
// ...
MyDel myDel = n => { var s = n + " Fanguzai!"; Console.WriteLine(s); };
myDel("Hi,");

异步 Lambda
通过 async 和 await 关键字,我们可以很简单并快速的创建包含异步处理的 lambda 表达式和语句。博主发表了约 8 篇关于异步的文章,你可以 点击进入目录 。
这里,我使用简单的异步调用方式,编写执行按钮触发的点击事件,即调用异步方法 DoAsync。

public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
} private async void button1_Click(object sender, EventArgs e)
{
await DoAsync();
} async Task DoAsync()
{
await Task.Delay(250);
}
}

现在,简化上面的的 Click 事件,并加上 async。

public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
button1.Click += async (sender, e) =>
{
await DoAsync();
};
} async Task DoAsync()
{
await Task.Delay(250);
}
}

在 LINQ 中使用 Lambda
许多 LINQ 中的参数都是一种委托类型的参数,如 Func<T, TResult>,可以定义输入参数以及返回类型。博主也发表了多篇关于 LINQ 的文章,你也可以 点击进入目录 。
public delegate TResult Func<TArg0, TResult>(TArg0 arg0)
Func<int, bool> 表示:int 为输入参数,bool 为返回值。
Func<int, int, bool> 表示:2个 int 为输入参数,一个 bool 为返回值。
示例:
Func<int, bool> myFunc = x => x == 250;
var result = myFunc(1314);
C# 的编译器可以自动推断输入参数的类型,即便是多个输入参数,当然,你也可以选择显式指定。
var nums = new[] { 2, 5, 0 };
var query = nums.Count(x => x > 2);
var query2 = nums.Count<int>(x => x < 2);
【备注】不要将 => 和 >= 搞错了,前者是 Lambda 运算符,后者是算术比较运算符。
Lambda 中的类型推断
编译器会根据 Lambda 主体、参数的委托类型以及 C# 语言规范和其它等一些因素,对我们所写的 Lambda 进行类型推断。

在这里,由于源数据是一个 int 数组,即我要查的数据为 IEnumerable<int> 类型,编译器在这里自动推断元素为 int 类型,意味着 Count 方法内的 x 你可以通过 “.” 在 VS 中显示对应 int 类型的属性和方法。
Lambda 中的变量使用范围
我们可以在 Lambda 的主体中引用范围之外的变量。如:
var nums = new[] { 2, 5, 0 }; //int[] 类型
var compareNum = 2.5;
var query = nums.Count(x => x == compareNum);
Lambda 的特点
Lambda 中包含输入参数的数量,必须与委托类型包含的参数数量一致。
Lambda 中的每个输入参数,必须都能够通过隐式转换为其对应的委托参数类型。
Lambda 中的返回值(如果有),必须能够隐式转换为委托的返回类型。
C# 知识回顾 - Lambda的更多相关文章
- [C#] C# 知识回顾 - Lambda
C# 知识回顾 - Lambda 序 它是第十一个希腊字母,一个拥有失意.无奈.孤独.低调等含义的流行符号,也指示一款称为“半条命”的游戏. 不过,这次我所讲的是 C# 中的 Lambda. 目录 L ...
- [.NET] C# 知识回顾 - 事件入门
C# 知识回顾 - 事件入门 [博主]反骨仔 [原文]http://www.cnblogs.com/liqingwen/p/6057301.html 序 之前通过<C# 知识回顾 - 委托 de ...
- [.NET] C# 知识回顾 - 委托 delegate (续)
C# 知识回顾 - 委托 delegate (续) [博主]反骨仔 [原文]http://www.cnblogs.com/liqingwen/p/6046171.html 序 上篇<C# 知识回 ...
- [C#] C# 知识回顾 - 委托 delegate
C# 知识回顾 - 委托 delegate [博主]反骨仔 [原文]http://www.cnblogs.com/liqingwen/p/6031892.html 目录 What's 委托 委托的属性 ...
- [C#] C# 知识回顾 - 表达式树 Expression Trees
C# 知识回顾 - 表达式树 Expression Trees 目录 简介 Lambda 表达式创建表达式树 API 创建表达式树 解析表达式树 表达式树的永久性 编译表达式树 执行表达式树 修改表达 ...
- C# 知识回顾 - 表达式树 Expression Trees
C# 知识回顾 - 表达式树 Expression Trees 目录 简介 Lambda 表达式创建表达式树 API 创建表达式树 解析表达式树 表达式树的永久性 编译表达式树 执行表达式树 修改表达 ...
- [C#] C# 知识回顾 - 你真的懂异常(Exception)吗?
你真的懂异常(Exception)吗? 目录 异常介绍 异常的特点 怎样使用异常 处理异常的 try-catch-finally 捕获异常的 Catch 块 释放资源的 Finally 块 一.异常介 ...
- [C#] C# 知识回顾 - 学会处理异常
学会处理异常 你可以使用 try 块来对你觉得可能会出现异常的代码进行分区. 其中,与之关联的 catch 块可用于处理任何异常情况. 一个包含代码的 finally 块,无论 try 块中是否在运行 ...
- [C#] C# 知识回顾 - 学会使用异常
学会使用异常 在 C# 中,程序中在运行时出现的错误,会不断在程序中进行传播,这种机制称为“异常”. 异常通常由错误的代码引发,并由能够更正错误的代码进行 catch. 异常可由 .NET 的 CLR ...
随机推荐
- Intellij idea 复制粘贴查找快捷键失效
遇到此问题,竟不能复制, 发现原因,是因为勾选了Vim模式, Tools,Vim Emulator,前面会有一个√,取消即可,如图: 我的是这个原因,复制粘贴快捷键失效,也有可能历史粘贴板的深度不够 ...
- OC学习11——循环引用与@class
转载自 OC学习篇之---@class关键字的作用以及#include和#import的区别 一.#import和#include的区别 当我们在代码中使用两次#include的时候会报错:因为#in ...
- MySQL 如何存储长度较大的varchar与blob
本文同时发表在https://github.com/zhangyachen/zhangyachen.github.io/issues/96 最近,在工作中遇到了MySQL中如何存储长度较长的字段类型问 ...
- 自定义bootstrap样式-9行样式自定义漂亮大气bootstrap导航栏
有人说前端发展太快,框架太多,各有所需,各有所长.看看这幅图,估计都知道这些框架,但是大部分公司中实际要用到的也就那么几个. 发展再快,框架再多.还是得回到原点,不就是Html+Css+JavaScr ...
- grep 与 find 简单命令
在使用linux的时候,经常会用到查找文件或者查找文本,下面介绍两个命令. grep 使用方法: // 在当前目录下递归查找class字符串 grep "string" -r . ...
- JVM虚拟机(一) 内存区域
JVM虚拟机内存组成: 如下图: 1. 程序计数器: (1)是一块较小的内存空间:可以看做当前程序执行子界面的行号指示器,字节码解析器执行的时候就是根据这个 ...
- NodeJS之微信开发
... 由于工作原因,太久没有写博客了,惭愧. 一直在说要买云服务器,前几天终于买了3个月,拿到之后就开始玩我最爱的nodejs,每天都是搞到12点1点过,乐此不疲啊. 从昨天开始玩的微信公众号,到今 ...
- Maven的下载、安装与环境配置
在创建一个项目时,搭建环境往往是编写具体代码的先决条件,而获取到所有需要的jar包是其中的重中之重.起初,人们在需要jar包的时候总会在网上四处查找,而且如果不知道某jar包版本的更迭,写出的代码或许 ...
- thinkinginjava学习笔记05_访问权限
Java中访问权限等级从大到小依次为:public.protected.包访问权限(没有关键词).private: 以包访问权限为界限,public.protected分别可以被任意对象和继承的对象访 ...
- thinkinginjava学习笔记04_初始化与清理
java沿用了c++的构造器,使用一个和类名完全一样的方法作为类的构造器,可以有多个构造器来通过不同的参数进行构造,称为重载:不仅是构造器可以重载,其他方法也一样通过不同的形参以及不同的返回值来实现重 ...