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,表名,检索字段,检索值和检索操作符,然后用户登陆后,通过自己权限来构 ...
随机推荐
- 【题解】Product
\(\color{brown}{Link}\) \(\text{Solution:}\) \(Question:\) \(\prod_{i=1}^n \prod_{j=1}^n \frac{lcm(i ...
- Java之微信支付(扫码支付模式二)案例实战
摘要:最近的一个项目中涉及到了支付业务,其中用到了微信支付和支付宝支付,在做的过程中也遇到些问题,所以现在总结梳理一下,分享给有需要的人,也为自己以后回顾留个思路. 一:微信支付接入准备工作: 首先, ...
- 使用Jest快照测试api
你知道什么很烦人吗?API不匹配. 有一天,后台开发人员在没有通知前端开发人员的情况下更改了其中一个api."我们认为dateCreated这个名字比created_at更好,"他 ...
- centos7 yum 安装nodejs、npm、cnpm、pm2、yarn
一.环境准备 1.1 查看系统环境 [root@localhost ~]# cat /etc/redhat-release CentOS Linux release 7.5.1804 (Core) [ ...
- ansible-playbook定义变量与使用
1. ansible-playbook变量定义与使用 命令行 在Inventory中定义 在Playbook中定义 在Role中定义 注册变量(register) 系统信息变量(facts) 2. 在 ...
- 风车签名 - 让管理APP变成一件简单的事儿
这是一款在Mac平台下安全可控的iOS签名管理软件,旨在对签名后的APP能够完全控制,包括APP的开启或禁用.设置到期时间锁.注入第三方动态库文件.设置安装限量.修改APP名称和自定义Bundle I ...
- Hello World -- 第一篇博客 -- 活着的意义
今年注定是不寻常的一年,因为技术,接触了许多大牛.通过一篇篇博文,看到了大牛们勤奋好学.孜孜不倦的精神,于是决定也开个博客,向大牛学习. 博客开了,写点什么呢?奈何肚子里墨水不多,吐出来也多是白沫,不 ...
- 记录一次源码扩展案列——FastJson自定义反序列化ValueMutator
背景:曾经遇到一个很麻烦的事情,就是一个json串中有很多占位符,需要替换成特定文案.如果将json转换成对象后,在一个一个属性去转换的话就出出现很多冗余代码,不美观也不是很实用. 而且也不能提前在j ...
- CSS常见反爬技术
目录 利用字体 反爬原理 应对措施 难点: 利用背景 反爬原理 应对措施 利用伪类 反爬原理 应对措施 利用元素定位 反爬原理 应对措施 利用字符切割 反爬原理 应对措施 利用字体 反爬原理 反爬原理 ...
- 【树形DP】CF 1293E Xenon's Attack on the Gangs
题目大意 vjudge链接 给n个结点,n-1条无向边.即一棵树. 我们需要给这n-1条边赋上0~ n-2不重复的值. mex(u,v)表示从结点u到结点v经过的边权值中没有出现的最小非负整数. 计算 ...