C# 表达式树讲解(一)
<h2>一、前言</h2><p>一直想写一篇Dpper的定制化扩展的文章,但是里面会设计到对Lambda表达式的解析,而解析Lambda表达式,就必须要知道表达式树的相关知识点。我希望能通过对各个模块的知识点或者运用能够多一点的讲解,能够帮助到园友了解得更多。虽然讲解得不全面,如果能成为打开这块的一把钥匙,也是蜗牛比较欣慰的。</p>
<p>表达式系列目录</p>
<p>C# 表达式树讲解(一)</p>
<p>C# 表达式树遍历(二)</p>
<p>C# 表达式树分页扩展(三)</p>
<p>C# 表达式树Lambda扩展(四)</p>
<h2>二、表达树理解</h2>
<p>表达式树以树形数据结构表示代码,其中每一个节点都是一种表达式,它将我们原来可以直接由代码编写的逻辑以表达式的方式存储在树状的结构里,从而可以在运行时去解析这个树,然后执行,实现动态的编辑和执行代码。在.Net 里面的Linq to SQL就是对表达式树的解析。</p>
<p>这里先讲解下表达式和表达式树,表达式相信大家都知道,比如x+5或者5,都可以算是表达式,而表达式树里面的树指的二叉树,也就是表达式的集合,C#中的Expression类就是表达式类。对于一棵表达式树,其叶子节点都是参数或者常数,非叶子节点都是运算符或者控制符。</p>
<h3>2.1、表达式的创建</h3>
<p>Lambda表达式方法:</p>
<div class="cnblogs_code">
<pre>Expression<Func<<span style="color: rgba(0, 0, 255, 1)">int</span>, <span style="color: rgba(0, 0, 255, 1)">int</span>,<span style="color: rgba(0, 0, 255, 1)">bool</span>>> fun = (x, y) => x < y</pre>
</div>
<p>这种方法创建出的表达式根节点类型为ExpressionType.Lambda,Type类型为返回值类型typeof(bool)</p>
<p>组装法(通过 API 创建表达式树):</p>
<div class="cnblogs_code">
<pre>ParameterExpression numParam = Expression.Parameter(<span style="color: rgba(0, 0, 255, 1)">typeof</span>(<span style="color: rgba(0, 0, 255, 1)">int</span>), "<span style="color: rgba(139, 0, 0, 1)">num</span>");
ConstantExpression five = Expression.Constant(5, <span style="color: rgba(0, 0, 255, 1)">typeof</span>(<span style="color: rgba(0, 0, 255, 1)">int</span>));
BinaryExpression numLessThanFive = Expression.LessThan(numParam, five);
Expression<Func<<span style="color: rgba(0, 0, 255, 1)">int</span>, <span style="color: rgba(0, 0, 255, 1)">bool</span>>> lambda1 =
Expression.Lambda<Func<<span style="color: rgba(0, 0, 255, 1)">int</span>, <span style="color: rgba(0, 0, 255, 1)">bool</span>>>(
numLessThanFive,
<span style="color: rgba(0, 0, 255, 1)">new</span> ParameterExpression[] { numParam });</pre>
</div>
<p>我们先创建了两个参数表达式num和5,然后用LessThan组装在一起,最终的表达式为“num<5”,expr的节点类型为LessThan,Type类型为typeof(bool)</p>
<p>我们先看看表达式树里面的构造</p>
<p>首先Expression<TDelegate>的功能是将强类型Lambda表达式表示为表达式树形式的数据结构,他的父类是LambdaExpression,比较他们代码可知,Lambda表达式的主体,名称和参数全部保存在LambdaExpression里面。</p>
<p>Expression<TDelegate>与LambdaExpression代码截图:</p>
<p><img style="display: inline; background-image: none" title="image" src="https://img2018.cnblogs.com/blog/1764554/201909/1764554-20190915071754702-320340030.png" alt="image" width="900" height="174" border="0"></p>
<p><img style="display: inline; background-image: none" title="image" src="https://img2018.cnblogs.com/blog/1764554/201909/1764554-20190915071755512-1913334399.png" alt="image" width="900" height="376" border="0"></p>
<p>LambdaExpression里面的Body就是我们的表达式。</p>
<p>C#表达式给我们提供了丰富的表达式类,进入到LambdaExpression类里面</p>
<p><img style="display: inline; background-image: none" title="image" src="https://img2018.cnblogs.com/blog/1764554/201909/1764554-20190915071757015-1549131452.png" alt="image" width="542" height="373" border="0"></p>
<p>方法返回类型以“Expression”结尾的,基本上都是一个表达式类。</p>
<p>每个表达式代表的定义和创建方法,可以参照微软官方文档https://docs.microsoft.com/zh-cn/dotnet/api/system.linq.expressions.binaryexpression?view=netframework-4.8</p>
<p>下面是平常使用最多的表达式</p>
<p><span style="color: rgba(0, 0, 255, 1)">ConstantExpression</span>:常量表达式</p>
<p><span style="color: rgba(0, 0, 255, 1)">ParameterExpression</span>:参数表达式</p>
<p><span style="color: rgba(0, 0, 255, 1)">UnaryExpression</span>:一元运算符表达式</p>
<p><span style="color: rgba(0, 0, 255, 1)">BinaryExpression</span>:二元运算符表达式</p>
<p><span style="color: rgba(0, 0, 255, 1)">TypeBinaryExpression</span>:is运算符表达式</p>
<p><span style="color: rgba(0, 0, 255, 1)">ConditionalExpression</span>:条件表达式</p>
<p><span style="color: rgba(0, 0, 255, 1)">MemberExpression</span>:访问字段或属性表达式</p>
<p><span style="color: rgba(0, 0, 255, 1)">MethodCallExpression</span>:调用成员函数表达式</p>
<p><span style="color: rgba(0, 0, 255, 1)">Expression<TDelegate>:</span>委托表达式</p>
<h3>2.2、表达式的解析</h3>
<p>表达式树解析</p>
<p>通过LambdaExpression类我们可以知道,表达式树包含:参数,表达式树类型,表达式,返回类型,Lambda表达式的委托以及Lambda表达式名称,如图所示:</p>
<p><img style="display: inline; background-image: none" title="image" src="https://img2018.cnblogs.com/blog/1764554/201909/1764554-20190915071757908-665869398.png" alt="image" width="900" height="240" border="0"></p>
<p>表达式解析:</p>
<p>所有的表达式都包含:左节点【Left】,右节点【Right】,类型【NodeType】,不同的表达式还会有其他属性,这里的左右节点依旧是表达式。</p>
<p>下图是BinaryExpression表达式截图</p>
<p><img style="display: inline; background-image: none" title="image" src="https://img2018.cnblogs.com/blog/1764554/201909/1764554-20190915071758702-744533449.png" alt="image" width="854" height="266" border="0"></p>
<p>表达式树和表达式里面的类型NodeType是一个枚举,一共有85个类型,有兴趣的朋友可以去了解下。</p>
<p>常用的类型如下:</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.And</span>:C#中类似于&</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.AndAlso</span>:C#中类似于&&</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.Or</span>:C#中类似于|</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.OrElse</span>:C#中类似于||</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.Equal</span>:C#中类似于==</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.NotEqual</span>:C#中类似于!=</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.GreaterThan</span>:C#中类似于></p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.GreaterThanOrEqual</span>:C#中类似于>=</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.LessThan</span>:C#中类似于<</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.LessThanOrEqual</span>:C#中类似于<=</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.Add</span>:C#中类似于+</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.AddChecked</span>:C#中类似于+</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.Subtract</span>:C#中类似于-</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.SubtractChecked</span>:C#中类似于-</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.Divide</span>:C#中类似于/</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.Multiply</span>:C#中类似于*</p>
<p><span style="color: rgba(0, 0, 255, 1)">ExpressionType.MultiplyChecked</span>:C#中类似于*</p>
<h3>2.3、编译表达式树</h3>
<p>在表达式创建那,我们组合创建了一个Lambda表达式,那么应该怎么使用它呢?在“表达式的解析”里面,LambdaExpression类和Expression<TDelegate>类都有一个Compile的方法,学名是Lambda表达式的委托,其实就是Lambda表达式编译函数的委托,所以我们只需要调用他,得到的结果就是一个函数方法。</p>
<p>代码修改如下:<br><br></p>
<div class="cnblogs_code">
<pre>ParameterExpression numParam = Expression.Parameter(<span style="color: rgba(0, 0, 255, 1)">typeof</span>(<span style="color: rgba(0, 0, 255, 1)">int</span>), "<span style="color: rgba(139, 0, 0, 1)">num</span>");
ConstantExpression five = Expression.Constant(5, <span style="color: rgba(0, 0, 255, 1)">typeof</span>(<span style="color: rgba(0, 0, 255, 1)">int</span>));
BinaryExpression numLessThanFive = Expression.LessThan(numParam, five);
Expression<Func<<span style="color: rgba(0, 0, 255, 1)">int</span>, <span style="color: rgba(0, 0, 255, 1)">bool</span>>> lambda1 =
Expression.Lambda<Func<<span style="color: rgba(0, 0, 255, 1)">int</span>, <span style="color: rgba(0, 0, 255, 1)">bool</span>>>(
numLessThanFive,
<span style="color: rgba(0, 0, 255, 1)">new</span> ParameterExpression[] { numParam });
Console.WriteLine($"<span style="color: rgba(139, 0, 0, 1)">Lambda的内容:{lambda1.ToString()}</span>");
<span style="color: rgba(0, 128, 0, 1)">//表达式的编译</span>
var func = lambda1.Compile();
Console.WriteLine($"<span style="color: rgba(139, 0, 0, 1)">Lambda的运行结果:{func(6)}</span>");
</pre>
</div>
<p>运行结果</p>
<p><img style="display: inline; background-image: none" title="image" src="https://img2018.cnblogs.com/blog/1764554/201909/1764554-20190915071759355-1661090116.png" alt="image" width="410" height="114" border="0"></p>
<h2>三、总结</h2>
<p>这里我们对表达式做了基本的讲解,相信大家对Lambda表达式有了初步的了解,下面我们将继续讲解对一个表达式树的遍历。</p>
</div>
<div id="MySignature" role="contentinfo">
<div style="font-size: 13px; border: 1px dashed rgb(45, 161, 45); padding: 10px 15px; background-color: rgb(248, 248, 248)">
<label style="font-weight: bold">
如果您觉得阅读本文对您有帮助,请点一下右下角推荐”按钮,博主在此感谢!另外您也可以选择【<strong>关注我</strong>】,可以很方便找到我!</label><br>
感谢您花时间阅读此篇文章,如果您觉得看了这篇文章之后心情还比较高兴,可以打赏一下,请博主喝上一杯咖啡,让博主继续码字……<br>
本文版权归作者和博客园共有,来源网址:https://www.cnblogs.com/snailblog 欢迎各位转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接
</div><br><br>
来源:https://www.cnblogs.com/snailblog/p/11521043.html
頁:
[1]