C# Lambda 知识回顾

时间:2022-05-15 08:12:19

C# Lambda 知识回顾

它是第十一个希腊字母,一个拥有失意、无奈、孤独、低调等含义的流行符号,也指示一款称为“半年命”的游戏。

不过,这次我所讲的是 c# 中的 lambda。

目录

  • lambda 简介
  • lambda 表达式
  • lambda 语句
  • 异步 lambda
  • 在 linq 中使用 lambda
  • lambda 中的类型推断
  • lambda 中的变量使用范围
  • lambda 的特点

lambda 简介

lambda 表达式,是一种简化的匿名函数,可用于创建委托或表达式目录树。其次,你也可以将 lambda 表达式作为参数进行传递,或者将它作用于函数调用值调用后返回的一个函数来使用。我们经常在 linq 中使用 lambda 表达式。

创建 lambda 表达式的简单语法形式:输入参数 => 表达式或语句块。其中,=> 为 lambda 运算符,可读作“goes to” 。

?
1
2
3
4
5
6
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
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
//
  // 摘要:
  //  基于谓词筛选值序列。
  //
  // 参数:
  // source:
  //  要筛选的 system.collections.generic.ienumerable<t>。
  //
  // predicate:
  //  用于测试每个元素是否满足条件的函数。
  //
  // 类型参数:
  // tsource:
  //  source 中的元素的类型。
  //
  // 返回结果:
  //  一个 system.collections.generic.ienumerable<t>,包含输入序列中满足条件的元素。
  //
  // 异常:
  // system.argumentnullexception:
  //  source 或 predicate 为 null。
  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 和构建表达式树,它也允许返回结果。

基本形式:( 输入参数 ) => 表达式 。

如:  

?
1
2
3
4
( ) => 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 语句和上面的 lambda 表达式相比,只是多了个大括号(“{ }”)。 

基本形式:( 输入参数 ) => { 表达式 } 。

lambda 语句的主体可以由任意数量的普通语句组成,不过,我们一般写的语句不多(三个左右吧)。

?
1
2
3
4
delegate void mydel(string s);
// ...
mydel mydel = n => { var s = n + " fanguzai!"; console.writeline(s); };
mydel("hi,");

异步 lambda

通过 async 和 await 关键字,我们可以很简单并快速的创建包含异步处理的 lambda 表达式和语句。

这里,我使用简单的异步调用方式,编写执行按钮触发的点击事件,即调用异步方法 doasync。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
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。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
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>,可以定义输入参数以及返回类型。

public delegate tresult func<targ0, tresult>(targ0 arg0)  

func<int, bool> 表示:int 为输入参数,bool 为返回值。

func<int, int, bool> 表示:2个 int 为输入参数,一个 bool 为返回值。

示例:

?
1
2
func<int, bool> myfunc = x => x == 250;
var result = myfunc(1314);

c# 的编译器可以自动推断输入参数的类型,即便是多个输入参数,当然,你也可以选择显式指定。

?
1
2
3
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 进行类型推断。

C# Lambda 知识回顾

在这里,由于源数据是一个 int 数组,即我要查的数据为 ienumerable<int> 类型,编译器在这里自动推断元素为 int 类型,意味着 count 方法内的 x 你可以通过 “.” 在 vs 中显示对应 int 类型的属性和方法。

lambda 中的变量使用范围

我们可以在 lambda 的主体中引用范围之外的变量。如:

?
1
2
3
var nums = new[] { 2, 5, 0 }; //int[] 类型
var comparenum = 2.5;
var query = nums.count(x => x == comparenum);

lambda 的特点

  • lambda 中包含输入参数的数量,必须与委托类型包含的参数数量一致。
  • lambda 中的每个输入参数,必须都能够通过隐式转换为其对应的委托参数类型。
  • lambda 中的返回值(如果有),必须能够隐式转换为委托的返回类型。

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持服务器之家!

原文链接:http://www.cnblogs.com/liqingwen/p/6216582.html