学习自

《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)的更多相关文章

  1. (转) Lambda表达式中的表达式lambda和语句lambda区别

    Lambda表达式可分为表达式lambda和语句lambda 表达式lambda:表达式位于 => 运算符右侧的lambda表达式称为表达式lambda (input parameters) = ...

  2. Lambda表达式中的表达式lambda和语句lambda区别

    Lambda表达式可分为表达式lambda和语句lambda 表达式lambda:表达式位于 => 运算符右侧的lambda表达式称为表达式lambda (input parameters) = ...

  3. Lambda表达式(lambda expression)⭐⭐⭐⭐⭐

    原作者 lambda表达式(lambda expression)实际上是匿名函数一种表示形式, 即没有函数名的函数:参数列表=>表达式或语句块,在我看来主要目是为了简化代码编写,提高代码可读性而 ...

  4. Python中特殊函数和表达式lambda,filter,map,reduce

    1.lambda:使用lambda表达式可以定义一个匿名函数 lambda表达式是一种简洁格式的函数.该表达式不是正常的函数结构,而是属于表达式的类型 (1)基本格式: lambda 参数,参数... ...

  5. Python学习教程(learning Python)--3.3 分支语句的条件表达式详解

    本节主要讨论分支语句的条件表达式问题. 在if或者if-else分支控制语句里由于都用到条件判断(表达式是真还是假),条件判断可以是一种关系运算也可以是布尔表达式. 本节将对if及if-else语句的 ...

  6. SQL 无限级分类语句

    原文:SQL 无限级分类语句 原表数据为: 此处用到了with关键字,在程序中也可以用递归实现,但觉得还是没有一条sql方便 with tb (ID,Name,ParentID,Sort) as( s ...

  7. day3_python之函数返回值、语句形式、表达式形式

    一. 函数对象 1. 函数是第一类对象,即函数可以当作数据传递 #1 可以被引用 #2 可以当作参数传递 #3 返回值可以是函数 #3 可以当作容器类型的元素 二.返回值 return的返回值没有类型 ...

  8. 点标记(lambda表达式+linq查询标记符)与linq语句(查询表达式)

    什么是Linq表达式?什么是Lambda表达式? 参照:https://www.cnblogs.com/zhaopei/p/5746414.html

  9. Java基础进阶:内部类lambda重点摘要,详细讲解成员内部类,局部内部类,匿名内部类,Lambda表达式,Lambda表达式和匿名内部类的区别,附重难点,代码实现源码,课堂笔记,课后扩展及答案

    内部类lambda重点摘要 内部类特点: 内部类可以直接访问外部类,包括私有 外部类访问内部类必须创建对象 创建内部对象格式: 外部类.内部类 对象名=new外部类().new内部类(); 静态内部类 ...

随机推荐

  1. linux c 编程 ------ 获取时间,计算程序执行时间

    #include <time.h> #include <stdio.h> #include <unistd.h> int main(int argc, char a ...

  2. Ajax跨域CORS

    在Ajax2.0中多了CORS允许我们跨域,但是其中有着几种的限制:Origin.Methods.Headers.Credentials 1.Origin 当浏览器用Ajax跨域请求的时候,会带上一个 ...

  3. bzoj千题计划229:bzoj4424: Cf19E Fairy

    http://www.lydsy.com/JudgeOnline/problem.php?id=4424 图是二分图的条件:没有奇环 所以,如果图不存在奇环,删除任意一条边都可以 如果存在奇环, 对于 ...

  4. AngularJS 启程二

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> </head> ...

  5. 求二叉树中第K层结点的个数

    一,问题描述 构建一棵二叉树(不一定是二叉查找树),求出该二叉树中第K层中的结点个数(根结点为第0层) 二,二叉树的构建 定义一个BinaryTree类来表示二叉树,二叉树BinaryTree 又是由 ...

  6. chrome 隐藏技能之 base64 图片转换

    有时候我们要转换图片为base64,或者将base64转回图片,可能都需要找一些在线工具或者软件类型的工具才行.当然 chrome 也算是软件,但是好在做前端的都有 chrome.好了,来看下简单的例 ...

  7. 【51Nod】1920 空间统计学 状压DP

    [题目]1920 空间统计学 [题意]给定m维空间中的n个点坐标,满足每一维坐标大小都在[0,3]之间,现在对于[0,3*m]的每个数字x统计曼哈顿距离为x的有序点对数.\(n \leq 2*10^5 ...

  8. Linux - trap 命令

    trap 命令用于指定在接收到信号后将要采取的动作,常见的用途是在脚本程序被中断时完成清理工作.当shell接收到sigspec指定的信号时,arg参数(命令)将会被读取,并被执行. trap 信号参 ...

  9. 安装odbc驱动

    1.下载对应的驱动 (32位/64位) http://www.oracle.com/technetwork/database/database-technologies/instant-client/ ...

  10. if语句引起的bug

    最近维护高手留下的api项目,客户端反馈一个bug过来,然后查找到可能出错的代码位置,是一个if语句,乍一看好像没什么问题,代码如下: if (company.UserId != userId || ...