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();
}

  创建表达式树:

  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 == ;
  (x) => x == ;
  (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 表达式和语句。博主发表了约 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(25);
}
}

  

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

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

在 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 == 2;
  var result = myFunc();

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

  var nums = new[] { , ,  };
  var query = nums.Count(x => x > );
  var query2 = nums.Count<int>(x => x < );

  【备注】不要将 => 和 >= 搞错了,前者是 Lambda 运算符,后者是算术比较运算符。

  

Lambda 中的类型推断

  编译器会根据 Lambda 主体、参数的委托类型以及 C# 语言规范和其它等一些因素,对我们所写的 Lambda 进行类型推断。

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

Lambda 中的变量使用范围

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

  var nums = new[] { , ,  };   //int[] 类型
  var compareNum = 2.5;
  var query = nums.Count(x => x == compareNum);

Lambda 的特点

  • Lambda 中包含输入参数的数量,必须与委托类型包含的参数数量一致。

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

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

C# 基础回顾系列

  《C# 知识回顾 - 委托 delegate》、《C# 知识回顾 - 委托 delegate (续)

  《C# 知识回顾 - 事件入门》、《C# 知识回顾 - Event 事件

  《string 与 String,大 S 与小 S 之间没有什么不可言说的秘密

  《C# 知识回顾 - 你真的懂异常(Exception)吗?

  《了解过入口函数 Main() 吗?带你用批处理玩转 Main 函数

  《C# 基础回顾 - 匿名方法

错误修正

  @likeheart :“半年命” -> “半条命”。详见评论区 10L。


【博主】反骨仔

【参考】微软官方文档

Lambda的更多相关文章

  1. 你知道C#中的Lambda表达式的演化过程吗?

    那得从很久很久以前说起了,记得那个时候... 懵懂的记得从前有个叫委托的东西是那么的高深难懂. 委托的使用 例一: 什么是委托? 个人理解:用来传递方法的类型.(用来传递数字的类型有int.float ...

  2. Linq表达式、Lambda表达式你更喜欢哪个?

    什么是Linq表达式?什么是Lambda表达式? 如图: 由此可见Linq表达式和Lambda表达式并没有什么可比性. 那与Lambda表达式相关的整条语句称作什么呢?在微软并没有给出官方的命名,在& ...

  3. 背后的故事之 - 快乐的Lambda表达式(一)

    快乐的Lambda表达式(二) 自从Lambda随.NET Framework3.5出现在.NET开发者眼前以来,它已经给我们带来了太多的欣喜.它优雅,对开发者更友好,能提高开发效率,天啊!它还有可能 ...

  4. Kotlin的Lambda表达式以及它们怎样简化Android开发(KAD 07)

    作者:Antonio Leiva 时间:Jan 5, 2017 原文链接:https://antonioleiva.com/lambdas-kotlin/ 由于Lambda表达式允许更简单的方式建模式 ...

  5. 9、委托、事件、Lambda

    开始 关于委托,肯定是要有问题的. 第一个问题,委托用来干什么? 看.net中的表述:在.net平台下,委托类型用来定义和相应应用程序中的回调.(回调?处理内存中两个实体双向通信的一种技术.)   第 ...

  6. java8中lambda表达式的应用,以及一些泛型相关

    语法部分就不写了,我们直接抛出一个实际问题,看看java8的这些新特性究竟能给我们带来哪些便利 顺带用到一些泛型编程,一切都是为了简化代码 场景: 一个数据类,用于记录职工信息 public clas ...

  7. 搭建一套自己实用的.net架构(3)续 【ORM Dapper+DapperExtensions+Lambda】

    前言 继之前发的帖子[ORM-Dapper+DapperExtensions],对Dapper的扩展代码也进行了改进,同时加入Dapper 对Lambda表达式的支持. 由于之前缺乏对Lambda的知 ...

  8. 如何设计一门语言(七)——闭包、lambda和interface

    人们都很喜欢讨论闭包这个概念.其实这个概念对于写代码来讲一点用都没有,写代码只需要掌握好lambda表达式和class+interface的语义就行了.基本上只有在写编译器和虚拟机的时候才需要管什么是 ...

  9. Speedment -- 利用lambda编写SQL

    众所周知Java8中加入了lambda语法,这一特性也帮助Java开发者极大的简化了开发.Speedment是一个利用lambda表达式操作数据库的框架,相比Java世界中现在非常流行的mybatis ...

随机推荐

  1. iframe用法

    <iframe src="http://caiyanli.top/" height="500"  width="500" frameb ...

  2. myeclipse学习总结一(在MyEclipse中设置生成jsp页面时默认编码为utf-8编码)

    1.每次我们在MyEclispe中创建Jsp页面,生成的Jsp页面的默认编码是"ISO-8859-1".在这种情况下,当我们在页面中编写的内容存在中文的时候,就无法进行保存.如下图 ...

  3. JavaScript Math和Number对象

    目录 1. Math 对象:数学对象,提供对数据的数学计算.如:获取绝对值.向上取整等.无构造函数,无法被初始化,只提供静态属性和方法. 2. Number 对象 :Js中提供数字的对象.包含整数.浮 ...

  4. JavaScript 自定义对象

    在Js中,除了Array.Date.Number等内置对象外,开发者可以通过Js代码创建自己的对象. 目录 1. 对象特性:描述对象的特性 2. 创建对象方式:对象直接量.new 构造函数.Objec ...

  5. JS与APP原生控件交互

    "热更新"."热部署"相信对于混合式开发的童鞋一定不陌生,那么APP怎么避免每次升级都要在APP应用商店发布呢?这里就用到了混合式开发的概念,对于电商网站尤其显 ...

  6. AFNetworking 3.0 源码解读(七)之 AFAutoPurgingImageCache

    这篇我们就要介绍AFAutoPurgingImageCache这个类了.这个类给了我们临时管理图片内存的能力. 前言 假如说我们要写一个通用的网络框架,除了必备的请求数据的方法外,必须提供一个下载器来 ...

  7. 【iOS】Xcode8+Swift3 纯代码模式实现 UICollectionView

    开发环境 macOS Sierra 10.12.Xcode 8.0,如下图所示: 总体思路 1.建立空白的storyboard用于呈现列表 2.实现自定义单个单元格(继承自:UICollectionV ...

  8. linux系统oracle-ora12505问题解决方案一

    说明:(1)Linux版本 Linux version 2.6.32.12-0.7-default (geeko@buildhost) (gcc version 4.3.4 [gcc-4_3-bran ...

  9. Linux基础介绍【第九篇】

    服务器添加3块磁盘的体系结构 [root@oldboylinux test]# free -m              total used free shared buffers cached M ...

  10. Markdown学习笔记

    分为两步: 1.阅读Markdown中文官网的文档 2.下载MarkdownPad2将中文官网中文档的例子敲一遍,其中Markdownpad2为官网中推荐的编辑器 备注: 如果只看中文官网文档,不边看 ...