Expression Trees(表达式树)是 C# 中一种将代码表示为数据结构的技术。它把 lambda 表达式转换成内存中的树形结构,而不是直接编译成可执行代码。这意味着你可以分析、修改或生成代码逻辑,在运行时动态处理表达式。
最常见的应用场景包括 LINQ to SQL 和 Entity Framework:它们利用表达式树将 C# 中的查询逻辑翻译成 SQL 语句。此外,表达式树也用于构建动态查询、AOP(面向切面编程)、Mock 框架(如 Moq)等高级场景。
表达式树的基本结构
在 C# 中,表达式树由 System.Linq.Expressions 命名空间下的类型构成。核心类是 Expression,它是所有表达式节点的基类。常见的派生类型有:
ParameterExpression:表示参数,比如xConstantExpression:表示常量,比如
5或
"hello"BinaryExpression:表示二元操作,如加法
+、等于
==UnaryExpression:表示一元操作,如取反
!MethodCallExpression:表示方法调用 LambdaExpression:表示整个 lambda 表达式
例如,lambda 表达式
x => x * 2被构造成一棵树: 根节点是 LambdaExpression 参数是 ParameterExpression(x) 主体是 BinaryExpression(乘法) 乘法的左右操作数分别是 x 和 ConstantExpression(2)
如何构建表达式树
手动构建表达式树可以更深入理解其结构。以下示例构建一个等效于
x => x > 5的表达式:
// 定义参数 ParameterExpression param = Expression.Parameter(typeof(int), "x"); <p>// 创建常量 ConstantExpression constant = Expression.Constant(5);</p><p>// 创建大于比较表达式 BinaryExpression body = Expression.GreaterThan(param, constant);</p><p>// 构建 lambda 表达式 Expression<Func<int, bool>> lambda = Expression.Lambda<Func<int, bool>>(body, param);</p>
此时,
lambda是一个表达式树对象,尚未执行。要执行它,需要编译:
Func<int, bool> compiled = lambda.Compile(); bool result = compiled(8); // 返回 true
这个过程分为“构造”和“编译执行”两个阶段,正是这种分离使得表达式可以被分析或转换。
解析表达式树
解析表达式树通常通过遍历其节点完成。你可以使用 ExpressionVisitor 类来实现自定义遍历逻辑。例如,你想提取表达式中所有的常量值:
public class ConstantExtractor : ExpressionVisitor
{
public readonly List<object> Constants = new List<object>();
<pre class='brush:php;toolbar:false;'>public override Expression Visit(Expression node)
{
if (node is ConstantExpression constExpr)
Constants.Add(constExpr.Value);
return base.Visit(node);
}}
使用方式:
Expression<Func<int, bool>> expr = x => x > 5 && x < 10; ConstantExtractor extractor = new ConstantExtractor(); extractor.Visit(expr); <p>// 输出:5 和 10 foreach (var c in extractor.Constants) Console.WriteLine(c);</p>
ExpressionVisitor 是抽象类,.NET 会自动递归访问表达式树的所有节点。你只需重写感兴趣的方法(如 VisitBinary、VisitMethodCall 等),就能实现精细控制。
表达式树的实际用途
除了 LINQ 查询翻译,表达式树还广泛用于:
动态查询构建:根据用户输入条件拼接查询,比如在 Web API 中实现灵活过滤 ORM 映射:Entity Framework 将表达式翻译为数据库命令 性能优化:相比反射,编译后的表达式树调用成员更快 DSL(领域特定语言)设计:用 C# 语法构造领域逻辑,再解析执行举个简单例子:动态创建属性访问器
public static Func<T, object> CreatePropertyGetter<T>(string propertyName)
{
var param = Expression.Parameter(typeof(T), "x");
var property = Expression.Property(param, propertyName);
var conversion = Expression.Convert(property, typeof(object));
var lambda = Expression.Lambda<Func<T, object>>(conversion, param);
return lambda.Compile();
}
调用
CreatePropertyGetter<person>("Name")</person> 可快速生成一个读取 Name 属性的委托,比反射快得多。
基本上就这些。表达式树本质是“代码即数据”,它让程序具备了检查和生成逻辑的能力,是 C# 元编程的重要工具。掌握它,能让你写出更灵活、更强大的框架级代码。
