Lambda的分类(语句Lambda和表达式Lambda)
学习自
《C#本质论》
Overview
在上一文中,我们简而又简的了解了一下,匿名方法和Lambda表达式,关于匿名方法这里暂且不表,本文我们来更加详细的了解一下Lambda表达式。
本文涉及到了大量的LInq方面的知识,如果看官你还没有接触到Linq那么本章可以略过。
语句Lambda和表达式Lambda
语句Lambda , 从语义上可以简单的理解为,一个语句块的委托。
比如说类似这样的,有多条语句组成的Lambda,被称为 语句Lambda
(x) =>{
int value =0;
for(int i =0 ; i<x ; i ++ ){
value+=1;
}
return value;
}
表达式 Lambda , 从语义上理解就是只有返回的表达式没有语句块。
比如说下面的,如果你接触过 Linq那么你一定很熟悉这种方式。
var result = persons.Where(t=>t.Name=="hello");
这么看起来没有什么特别显著的区别,请继续向下看。
表达式树(Expression tree)
语句Lambda和表达式Lambda最后的生成的结果,是完全不同的,一个语句Lambda将会生成一个委托,而一个表达式Lambda既可以生成一个委托,也可以转换为表达式树(expression tree)。 表达式树是一个对象,允许编译器对Lambda表达式的保主体进行分析。
在ORM框架中常常会见到类似这样的代码
var result = db.Persons.Where(t=> t.Name=="Hello");
假设Person是数据库中的一个表 假设表中有大概10w条数据,然后我们要再这个表中找到 Name = 'Hello' 的Person
当我们执行查询的时候,有两个解决思路
- 将数据库所有的数据都返回给客户端,然后客户端为灭一条记录创建一个对象,形成一个集合,然后我们通过定义的委托,来判断是否是我们需要的对象,如果是我们需要的对象那么拿出来,其他的对象丢弃掉。这种方式理论上是可行的,但是,从一个数据表中插叙1条或者几条记录,就要将所有的数据都吃进内存中,后果可想而知,所以这种方式可想而知。
- 第二个解决思路, 将我们“Lambda” 发送给您服务器数据库,然后让服务器数据库进行过滤操作,然后我们仅仅拿到符合我们的结果的几条数据。这种方式使我们ORM所通用的解决思路。
当然仅仅直接将我们是Lambda表达式发送给数据库,显然是不行的,所以我们要在中间做一层转换,将我们的Lambda表达式转换为描述Lambda描述的对象表达式树 ,而不是一个编译好的一个匿名函数的代码。应为表达式数代表着是数据而不是编译好的代码,所我们能在运行的时候去分析Lambda,在ORM框架中就将 Where() 方法中接收到的表达式树转换为Sql查询语句 交给数据库去执行,并返回我们需要的数据。
转换如下:

NOTE: 表达式不是仅仅只能转换为sql语句,而是通过一个表达式树计算程序(evaluator) ,将表达式转换为任意的查询语言。
语句Lambda和表达式Lambda是如何区分的
不管是将Lambda转换为委托哈市表达式树,一个Lambda表达式都会在编译时进行全面的语义解析,如果是转换为委托,那么就会将这个Lambda 生成为一个方法,并创建委托。但是在使用Linq的时候明显不是这样的,编译器是如何判断一个Lambda该如何处理?生成委托,还是生成表达式树?
这里将以Where方法为例子,进行讲解。
利用Where方法查询集合:
static List<Person> personList = new List<Person>
{
new Person() { Name="1", Gender="Male", Age="12"},
new Person() { Name="2", Gender="Male", Age="12"},
new Person() { Name="3", Gender="Male", Age="12"},
};
static void Main(string[] args)
{
personList.Where(t => t.Name.Equals("1"));
}
F12转到Where方法的定义 , 我们发现他是一个IEnumerable 泛型的 的扩展方法。
public static IEnumerable<TSource> Where<TSource>(this IEnumerable<TSource> source, Func<TSource, bool> predicate);
看一下反编译

在这里是生成了委托。
LINQ 中的Where方法,
static void Main(string[] args)
{
DBDataContext db = new DBDataContext();
var result = db.Roles.Where(t => t.RoleName == "TestRole");
}
是IQueryable 接口的扩展方法
public static IQueryable<TSource> Where<TSource>(this IQueryable<TSource> source, Expression<Func<TSource, bool>> predicate);
查看一下反编译,没有生成委托,而是生成了一个 Linq 表达式。

- 当Where() 方法的调用者能够隐式转换为 IEnumerable 类型时,Lambda表达式将会生成委托
- 当Where() 方法的调用者能够隐式转换为IQueryable 类型时,Lambda表达式将会生成表达式树
结语
说实话,本文确实不太好,强烈推荐大家去看一下,原文,本文是《C#》本质论的学习笔记。
Lambda的分类(语句Lambda和表达式Lambda)的更多相关文章
- (转) Lambda表达式中的表达式lambda和语句lambda区别
Lambda表达式可分为表达式lambda和语句lambda 表达式lambda:表达式位于 => 运算符右侧的lambda表达式称为表达式lambda (input parameters) = ...
- Lambda表达式中的表达式lambda和语句lambda区别
Lambda表达式可分为表达式lambda和语句lambda 表达式lambda:表达式位于 => 运算符右侧的lambda表达式称为表达式lambda (input parameters) = ...
- Lambda表达式(lambda expression)⭐⭐⭐⭐⭐
原作者 lambda表达式(lambda expression)实际上是匿名函数一种表示形式, 即没有函数名的函数:参数列表=>表达式或语句块,在我看来主要目是为了简化代码编写,提高代码可读性而 ...
- Python中特殊函数和表达式lambda,filter,map,reduce
1.lambda:使用lambda表达式可以定义一个匿名函数 lambda表达式是一种简洁格式的函数.该表达式不是正常的函数结构,而是属于表达式的类型 (1)基本格式: lambda 参数,参数... ...
- Python学习教程(learning Python)--3.3 分支语句的条件表达式详解
本节主要讨论分支语句的条件表达式问题. 在if或者if-else分支控制语句里由于都用到条件判断(表达式是真还是假),条件判断可以是一种关系运算也可以是布尔表达式. 本节将对if及if-else语句的 ...
- SQL 无限级分类语句
原文:SQL 无限级分类语句 原表数据为: 此处用到了with关键字,在程序中也可以用递归实现,但觉得还是没有一条sql方便 with tb (ID,Name,ParentID,Sort) as( s ...
- day3_python之函数返回值、语句形式、表达式形式
一. 函数对象 1. 函数是第一类对象,即函数可以当作数据传递 #1 可以被引用 #2 可以当作参数传递 #3 返回值可以是函数 #3 可以当作容器类型的元素 二.返回值 return的返回值没有类型 ...
- 点标记(lambda表达式+linq查询标记符)与linq语句(查询表达式)
什么是Linq表达式?什么是Lambda表达式? 参照:https://www.cnblogs.com/zhaopei/p/5746414.html
- Java基础进阶:内部类lambda重点摘要,详细讲解成员内部类,局部内部类,匿名内部类,Lambda表达式,Lambda表达式和匿名内部类的区别,附重难点,代码实现源码,课堂笔记,课后扩展及答案
内部类lambda重点摘要 内部类特点: 内部类可以直接访问外部类,包括私有 外部类访问内部类必须创建对象 创建内部对象格式: 外部类.内部类 对象名=new外部类().new内部类(); 静态内部类 ...
随机推荐
- net.sf.json------json解析
下载地址 [plain] view plain copy 本次使用版本:http://sourceforge.net/projects/json-lib/files/json-lib/json-l ...
- python学习笔记6--mockserver
一.mockserver的应用 有时候测试我们需要调用一些三方接口或者未开发完成的接口,完成我们的业务流程测试,但是这时候可能我们只知道接口返回值,接口并没有完全开发完成或可以让我们任意调用,这时候就 ...
- iOS编码规范参考
目录 注释 1.1 多行注释 1.2 单行注释 1.3 函数的注释 命名 2.1 常量的命名 2.2 函数的命名 2.3 变量的命名 2.3.1 成员变量 2.3.2 公 ...
- Anaconda+django写出第一个web app(九)
今天来学习外键的使用,用外键来连接数据库中的两个表. 当我们的tutorials非常多的时候,目前的显示方式就会使得页面非常凌乱.我们可以考虑把这些教程分为不同的系列,页面只显示标题以及概要等信息,进 ...
- Zookeeper命名服务——生成分布式有序且唯一id
生成分布式有序且唯一id的方法有很多种,使用zookeeper是比较简单的一种方法,只是生成的速度不高,这里只是一个借助zk的版本号生成分布式唯一且有序id的例子. ZkIdGenerator.jav ...
- 基于滑动窗口协议写的程序(UDP实现) .
正好有一个大作业关于用socket实现滑动窗口协议,所以写了一个,模拟接收方与发送方窗口都是2,用两个线程实现. 下面是代码,注释的比较详细了. socket_udp.h #include<st ...
- linux网桥浅析
linux网桥浅析 原文链接:http://hi.baidu.com/_kouu/item/25787d38efec56637c034bd0 什么是桥接?简单来说,桥接就是把一台机器上的若干个网络接口 ...
- Java工具库:
1. 重试框架: https://docs.spring.io/spring-batch/trunk/reference/html/retry.html <dependency> < ...
- vs code 的背景颜色主题还有背景图片的自定义方法
先说颜色主题吧: 依次点击文件--->首选项---->颜色主题 你就可以看到不同的颜色主题了 如果你还觉得不好看,不满意,不符合你的审美风格 你还可以在插件库里面下载相关插件: THEME ...
- 【Android】隐藏底部虚拟按键
Google的官方文档是: https://developer.android.com/training/system-ui/navigation.html#behind 示例代码 1 View de ...