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> 的参数。该方法有多个重载,这里只列举了其中一个。

//
  // 摘要:
  //  基于谓词筛选值序列。
  //
  // 参数:
  // 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 和构建表达式树,它也允许返回结果。

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

如:  

  ( ) => 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 语句的主体可以由任意数量的普通语句组成,不过,我们一般写的语句不多(三个左右吧)。

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

异步 Lambda

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

这里,我使用简单的异步调用方式,编写执行按钮触发的点击事件,即调用异步方法 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>,可以定义输入参数以及返回类型。

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 中的返回值(如果有),必须能够隐式转换为委托的返回类型。

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

(0)

相关推荐

  • C# Lambda表达式用途深入理解

    假如我们想要从一个整型数组中取出其中是奇数的选项,其实现方式有很多,我们通过下面三种实现方式来对对比理解Lambda表达式的用途 方法一:命名方法 复制代码 代码如下: public class Common { public delegate bool IntFilter(int i); public static List<int> FilterArrayOfInt(int[] ints, IntFilter filter) { var lstOddInt = new List<in

  • C#基础之Lambda表达式用法实例教程

    本文以实例形式介绍了C#中Lambda表达式的用法,分享给大家供大家参考之用.具体如下: 从委托的角度来看,Lambda表达式与匿名方法没有区别.在前面C#基础之匿名方法一文中,我们使用了匿名方法来调用List<T>的FindAll方法.从C# 3.0开始,在使用匿名方法的地方,完全可以用Lambda表达式来代替.Lambda表达式的定义方式为:"([参数列表]) => 表达式".运算符"=>"是一种与赋值运算"="具有相

  • C#中使用Lambda表达式自定义比较器实现两个列表合并实例

    一次项目有这样的需求,本地存储了json数据,可以转化为对应的List列表,现在需要更新,从服务器那里获取最新的数据更改.总的来说就是本地有个List表,如果数据需要更新,则会向服务器发送请求来获取需要更改的部分List表格,然后客户端这边就要去处理合并完整这次的更新,弄了一个下午,终于搞定,这次来mark一下. 主要推送的数据的两个特性: 1.如果之前的列表需要增加,则出现在最新的数据推送List中 2.如果数据需要修改的话,则一样推送更新,保持id不变,内容有所改变 示例代码如下图: 复制代

  • C#特性之匿名方法和Lambda表达式

    在我们程序中,经常有这样一些需求: 1.       需要一个临时方法,这个方法只会使用一次,或者使用的很少. 2.       这个方法的方法体很短,以至于比方法声明都短,写起来实在没劲(我将其称之为"一句话方法"). 没办法,这样的方法写起来真是吃力不讨好,比如一些按钮事件处理中,有些按钮点击就是弹出一个对话框,或者调用一下别的什么方法.比如下面的代码: 复制代码 代码如下: this.btnRefresh.Click += new System.EventHandler(this

  • C#用Lambda和委托实现模板方法

    1 问题描述查看下面这段代码: 复制代码 代码如下: int[] a = [1,2,3]; for (int i =0; i<a.length; i++)  {    a[i] = a[i] * 2;  } for (int i =0; i<a.length; i++)  {    Console.WriteLine(a[i]);  } 很明显,上述代码中存在for循环的重复代码. 2 解决方案如何消除重复?使用委托. •定义委托 复制代码 代码如下: delegate int mapfun(

  • C#3.0中Lambda表达式详解

    在C#2.0中,微软给我们带来了一些新的特性,例如泛型,匿名委托等.然而,这些新的特性多多少少会给人一种从别的语言中"抄"来的感觉(例如泛型类似C++的模板,一些特性类似Java中的一些东西).但是在C#3.0中,微软给我带来的一些新特性可能是以前所有开发语言都没有的特性.这无疑大大的体现了C#3.0在开发语言中强大的优势. Lambda表达式 Lambda 表达式是一个匿名函数,它可以包含表达式和语句,并且可用于创建委托或表达式目录树类型.所有 Lambda 表达式都使用 Lambd

  • 理解C#中的Lambda表达式

    先来看两段代码: 复制代码 代码如下: Thread t = new Thread(() => { AddIt AddDelegate = new AddIt(AddItem); this.Invoke(AddDelegate); }); Thread t3 = new Thread(new ThreadStart(() => { AddIt AddDelegate = new AddIt(AddItem); this.Invoke(AddDelegate); })); 这两种写法都是可以的,

  • 初步认识C#中的Lambda表达式和匿名方法

    写在前面 元旦三天在家闲着无事,就看了看Linq的相关内容,也准备系统的学习一下,作为学习Linq的前奏,还是先得说说Lambda与匿名方法的知识点.也算是对知识点的查漏补缺吧,也许你会说这没啥大不了的,项目中都在用,但是有些知识,你回头在查看的时候,总会有那么点不一样的收获,这点我是感同身受的,我看书有个习惯,一本书,我能看个三四遍,每次总会有收获.当然,你可以说,当时肯定没认真看,不是那样子的,我认为最直接的原因在于,当时你看是看,没有在真正的项目中遇到过,所以你心里对它的理解并不深,如果在

  • C# 中如何利用lambda实现委托事件的挂接

    委托定义如下: 复制代码 代码如下: public class SocketSp{ public delegate void ReceiveCompleted(byte[] receiveBuffer, int receiveTotalLen,Exception ex); public ReceiveCompleted receiveCompleted;} 挂接方定义如下 复制代码 代码如下: public class LinkOuter{ SocketSp linkOuterSocket =

  • C#中方法的直接调用、反射调用与Lambda表达式调用对比

    想调用一个方法很容易,直接代码调用就行,这人人都会.其次呢,还可以使用反射.不过通过反射调用的性能会远远低于直接调用--至少从绝对时间上来看的确是这样.虽然这是个众所周知的现象,我们还是来写个程序来验证一下.比如我们现在新建一个Console应用程序,编写一个最简单的Call方法. 复制代码 代码如下: class Program {     static void Main(string[] args)     {             } public void Call(object o

随机推荐