c# 表达式树(一)
前言
打算整理c# 代码简化史系列,所以相关的整理一下,简单的引出一下概念。
什么是表达式树呢?
表达式树以树形数据结构表示代码,其中每一个节点都是一种表达式,比如方法调用和 x < y 这样的二元运算等。
这个是什么意思呢?用结构表示代码? 用静态的表示动态的,一般来说是某种约定。
比如计算机中的强弱电路,可能这样不好理解。举一个盒子的例子:

假设我要计算加法,那么如果表示加法呢?我用一个盒型结构,把第一个数放在第一个位置,把第二个数放在第二个位置,然后第三个位置我传入方法,表示第一个和第二个会执行第三个位置的方法,在这里呢,还是结构,因为并没有去运行,只是说组合了这样一种结构。
现在呢,假设按照某种约定组合成一种结构,那么这种就称为表达式,就是用来表示某种情况的嘛。然后呢,现在这种表达式是树,那么就叫表达式树了。
这里介绍一下表达式,来增强一波:

然后再来透析一波:

正文
用一个例子来表示正则表达吧,例子是官网的,但是官网解释的比较含糊,所以再来解释一波吧。
官网用的一个例子是:Where(company => (company.ToLower() == "coho winery" || company.Length > 16))
那么来看一下吧:
string[] companies = { "Consolidated Messenger", "Alpine Ski House", "Southridge Video", "City Power & Light",
"Coho Winery", "Wide World Importers", "Graphic Design Institute", "Adventure Works",
"Humongous Insurance", "Woodgrove Bank", "Margie's Travel", "Northwind Traders",
"Blue Yonder Airlines", "Trey Research", "The Phone Company",
"Wingtip Toys", "Lucerne Publishing", "Fourth Coffee" };
// The IQueryable data to query.
IQueryable<String> queryableData = companies.AsQueryable<string>();
有一个数组,然后转换成IQueryable 格式,这么做的目的其实就是因为queryable 实现了一些expression的属性。

好吧,暂时就不解释这几个参数的作用,后面看下去自然就明白了。
接着放代码:
ParameterExpression pe = Expression.Parameter(typeof(string), "company");
这个意思就是说创建了一个属性是company的变量,相当于我们以前的xy,名字随便取。
Expression left = Expression.Call(pe, typeof(string).GetMethod("ToLower", System.Type.EmptyTypes));
Expression right = Expression.Constant("coho winery");
Expression e1 = Expression.Equal(left, right);
因为其实树结构,那么这里的参数left就是左子树,right 就是右子树。
left 呢,这个call 就是说pe(也就是company变量)将会执行一个方法,ToLower,对应的就是company.ToLower()。
然后右边就是一个固定的参数coho winery,现在的表达式就是company.ToLower()=='coho winery',返回的是一个bool类型。
left = Expression.Property(pe, typeof(string).GetProperty("Length"));
right = Expression.Constant(16, typeof(int));
Expression e2 = Expression.GreaterThan(left, right);
接下来就是就是获取compay的属性Length,然后和int 类型相比,就是conpany.length>16
Expression predicateBody = Expression.OrElse(e1, e2);
那么就是e1和e2相连,中间用的是or,company.ToLower() == "coho winery" || company.Length > 16 好的现在表达式完了,那么如何和数据联系在一起呢?
// Create an expression tree that represents the expression
// 'queryableData.Where(company => (company.ToLower() == "coho winery" || company.Length > 16))'
MethodCallExpression whereCallExpression = Expression.Call(
typeof(Queryable),
"Where",
new Type[] { queryableData.ElementType },
queryableData.Expression,
Expression.Lambda<Func<string, bool>>(predicateBody, new ParameterExpression[] { pe }));
// ***** End Where *****
// ***** OrderBy(company => company) *****
// Create an expression tree that represents the expression
// 'whereCallExpression.OrderBy(company => company)'
MethodCallExpression orderByCallExpression = Expression.Call(
typeof(Queryable),
"OrderBy",
new Type[] { queryableData.ElementType, queryableData.ElementType },
whereCallExpression,
Expression.Lambda<Func<string, string>>(pe, new ParameterExpression[] { pe }));
// ***** End OrderBy *****
// Create an executable query from the expression tree.
IQueryable<string> results = queryableData.Provider.CreateQuery<string>(orderByCallExpression);
whereCallExpression 和 orderByCallExpression 表示要执行的操作,用whereCallExpression 举例目标类型是Queryable,调用where,然后表达式是predicateBody,参数是pe。
orderByCallExpression 类推。
最后一步就是传递表达:IQueryable results = queryableData.Provider.CreateQuery(orderByCallExpression);
里面的实现是非常复杂的,我自己也没有去看,因为觉得没有必要,这种东西就是一个工具,谁要是这样写,那可正是思维逻辑不是一般的强,一般来说和汇编差不多。
需要明白的就是它不会立即去执行,而是就是一个表达式和其紧密连接。有兴趣可以去了解iqueryable的实现,复杂的一批。
那么不管其多么复杂,就是本质上就是制定一套规则,我们按照它这个规则然后给我们填充,那么就会对应相应的结果给我们,我们可能设计不出这么好的表达式,但是有时候我们也会去制作相应的规则,比如说某种格式等,但你不要去想想它的代码多优雅,因为其稳定性很高,不需要追求优雅。
那么我们为了延后实现,我们就要去这样做吗?如果这样做的话,我想很多人会设计出另外一套,没有这么繁琐,可能就是几个参数,然后一个委托组合成一颗树,虽然很大的局限性,但是写这样的代码真的痛苦。
这个时候人们就想有没有什么能中间转换一下的呢?比如说我写一串字符,然后我就自动按照某种规则去解析不就可以了,但是这种有一个很不好的地方在于,字符串是弱类型调试相当麻烦,这时候就瞄准好了lambda了。
若 lambda 表达式被分配给 Expression<TDelegate> 类型的变量,则编译器可以发射代码以创建表示该 lambda 表达式的表达式树。
C# 编译器只能从表达式 Lambda(或单行 Lambda)生成表达式树。 它无法解析语句 lambda (或多行 lambda)。
举个例子:
Expression<Func<int, bool>> lambda = num => num < 5;
就可以使用lambda表达式进行一个expression的转换。
从Expression到Expression 之间呢,还有一层,他们的继承关系是
Object
Expression
LambdaExpression
Expression<TDelegate>
很多时候lambda 表达式转换的表达式就可以为我们解决大部分问题,但是不要觉得这是Expression的全部,因为转换的只有一行,然后expression还有很多是无法用lambda来表示的。
续
未完待续。
c# 表达式树(一)的更多相关文章
- 再讲IQueryable<T>,揭开表达式树的神秘面纱
接上篇<先说IEnumerable,我们每天用的foreach你真的懂它吗?> 最近园子里定制自己的orm那是一个风生水起,感觉不整个自己的orm都不好意思继续混博客园了(开个玩笑).那么 ...
- [C#] C# 知识回顾 - 表达式树 Expression Trees
C# 知识回顾 - 表达式树 Expression Trees 目录 简介 Lambda 表达式创建表达式树 API 创建表达式树 解析表达式树 表达式树的永久性 编译表达式树 执行表达式树 修改表达 ...
- 轻量级表达式树解析框架Faller
有话说 之前我写了3篇关于表达式树解析的文章 干货!表达式树解析"框架"(1) 干货!表达式树解析"框架"(2) 干货!表达式树解析"框架" ...
- 用五分钟重温委托,匿名方法,Lambda,泛型委托,表达式树
这些对老一代的程序员都是老生常谈的东西,没什么新意,对新生代的程序员却充满着魅力.曾经新生代,好多都经过漫长的学习,理解,实践才能掌握委托,表达式树这些应用.今天我尝试用简单的方法叙述一下,让大家在五 ...
- LinqToDB 源码分析——处理表达式树
处理表达式树可以说是所有要实现Linq To SQL的重点,同时他也是难点.笔者看完作者在LinqToDB框架里面对于这一部分的设计之后,心里有一点不知所然.由于很多代码没有文字注解.所以笔者只能接合 ...
- LinqToDB 源码分析——生成表达式树
当我们知道了Linq查询要用到的数据库信息之后.接下就是生成对应的表达式树.在前面的章节里面笔者就已经介绍过.生成表达式树是事实离不开IQueryable<T>接口.而处理表达式树离不开I ...
- 干货!表达式树解析"框架"(1)
最新设计请移步 轻量级表达式树解析框架Faller http://www.cnblogs.com/blqw/p/Faller.html 关于我和表达式树 其实我也没有深入了解表达式树一些内在实现的原理 ...
- 干货!表达式树解析"框架"(2)
最新设计请移步 轻量级表达式树解析框架Faller http://www.cnblogs.com/blqw/p/Faller.html 为了过个好年,我还是赶快把这篇完成了吧 声明 本文内容需要有一定 ...
- 干货!表达式树解析"框架"(3)
最新设计请移步 轻量级表达式树解析框架Faller http://www.cnblogs.com/blqw/p/Faller.html 这应该是年前最后一篇了,接下来的时间就要陪陪老婆孩子了 关于表达 ...
- Lind.DDD.ExpressionExtensions动态构建表达式树,实现对数据集的权限控制
回到目录 Lind.DDD框架里提出了对数据集的控制,某些权限的用户为某些表添加某些数据集的权限,具体实现是在一张表中存储用户ID,表名,检索字段,检索值和检索操作符,然后用户登陆后,通过自己权限来构 ...
随机推荐
- 00 在Windows环境中开发Cordova项目的准备工作
1.开发环境准备: 安装nodejs 安装Cordova 安装Visual Studio Code 安装nodejs步骤: 通过nodejs官网(https:/ ...
- 源码安装IVRE
简介:IVRE(又名DRUNK)是一款开源的网络侦查框架工具,IVRE使用Nmap.Zmap进行主动网络探测.使用Bro.P0f等进行网络流量被动分析,探测结果存入数据库中,方便数据的查询.分类汇总统 ...
- JVM系列【3】Class文件加载过程
JVM系列笔记目录 虚拟机的基础概念 class文件结构 class文件加载过程 jvm内存模型 JVM常用指令 GC与调优 Class文件加载过程 JVM加载Class文件主要分3个过程:Loadi ...
- linxu 命令
top | grep java 统计 java 进程使用的资源比率 nohub java -jar test.war & 后台运行 test.war 程序,标准输出到 test.war 程序目 ...
- git克隆指定分支到本地
我们每次使用命令 git clone https://xxx.com/android-app.git 默认 clone 的是这个仓库的 master 分支. 使用Git下载指定分支命令为:git cl ...
- volatile、ThreadLocal的使用场景和原理
并发编程中的三个概念 原子性 一个或多个操作.要么全部执行完成并且执行过程不会被打断,要么不执行.最常见的例子:i++/i--操作.不是原子性操作,如果不做好同步性就容易造成线程安全问题. 可见性 多 ...
- 多测师讲解接口 _需求文档(用户增删改查)_高级讲师肖sir
首先连接Duoceshi_new网络 密码为Duoceshi_new,因为接口项目部署在Duoceshi_new网段中. 测试工具:postman域名:http://192.168.1.2:8081/ ...
- vs code 编译python 输出到调试控制台
如图所示,在debug菜单中点击齿轮按钮,进入launch.json,更改console选项的值(有三种) "console": "internalConsole&quo ...
- day20 Pyhton学习 面向对象-类与类之间的关系
一.类与类之间的依赖关系 class Elphant: def __init__(self, name): self.name = name def open(self, ref): print(&q ...
- CSP-S2020AFO记
2020-10.11 考初赛辣. 选择题考了一堆时间复杂度,一个不会(卒) 我寻思这01背包哪里能用贪心? 啊,这,这,这手写快排竟如此简单,手写取Max,手写队列,两个字符串颠来倒去,竟活到爆! 震 ...