EF Core表达式树

embedded/2025/4/2 6:34:50/

文章目录

  • 前言
  • 一、表达式树与委托的区别
  • 二、动态构建表达式树
    • 示例1
    • 示例2
    • 示例3
    • 高级技巧:表达式合并
  • 三、ExpressionTreeToString
    • 安装方法
    • 基本用法
    • 支持的格式化风格
  • 四、注意事项
  • 总结


前言

在 Entity Framework Core 中,表达式树(Expression Tree) 是 LINQ 查询的核心机制,它允许将 C# 代码中的查询逻辑转换为 SQL 语句,从而在数据库服务器端高效执行。

一、表达式树与委托的区别

  1. 委托(如 Func<T, bool>
    直接编译为可执行的代码,运行时在内存中过滤数据(客户端评估)。

    Func<House, bool> func = p => p.Owner.Contains("tom");
    var res=dbContext.Houses.Where(exp1).ToList(); // 在客户端过滤!
    
  2. 表达式树(如 Expression<Func<T, bool>>
    保持查询逻辑的抽象语法树结构,EF Core 可将其转换为 SQL服务器端评估)。

    Expression<Func<House, bool>> exp1 = b => b.Owner.Contains("tom");
    var res=dbContext.Houses.Where(exp1).ToList();//生成 SQL:WHERE Owner like '%tom%'
    

二、动态构建表达式树

  1. 当需要根据运行时条件动态生成查询时,手动构建表达式树非常有用。
  2. ParameterExpressionBinaryExpressionMethodCallExpressionConstantExpression等类几乎都没有提供构造方法,而且所有属性也几乎都是只读,因此我们一般不会直接创建这些类的实例,而是调用Expression类的ParameterMakeBinaryCallConstant等静态方法来生成,这些静态方法我们一般称作创建表达式树的工厂方法,而属性则是通过方法参数类设置。
  3. 工厂方法
    加法:Add
    短路与运算:AndAlso
    数组元素访问:ArraryAccess
    方法访问:Call
    三元条件运算符:Condition
    常量表达式:Constant
    类型转换:Convert
    大于运算符:GreaterThan
    小于运算:LessThan
    大于或等于运算符:GreaterThanOrEqual
    创建二元运算:MakeBinary
    不等于运算:NotEqual
    短路或运算:OrElse
    表达式的参数:Parameter

示例1

  1. 动态过滤Owner包含关键字

    using System.Linq.Expressions;
    using (MyDBContext dbContext=new MyDBContext())
    {string name = Console.ReadLine();// 参数表达式:代表实体对象(如 p => ... 中的 p)ParameterExpression param = Expression.Parameter(typeof(House), "p");// 属性访问:p.OwnerMemberExpression nameProperty = Expression.Property(param, "Owner");MethodInfo containsMethod = typeof(string).GetMethod("Contains", new[] { typeof(string) });// 参数值ConstantExpression keywordConstant = Expression.Constant(name);MethodCallExpression nameCondition = Expression.Call(nameProperty, containsMethod, keywordConstant);// 组合为 Lambda 表达式Expression<Func<House, bool>> expr =Expression.Lambda<Func<House, bool>>(nameCondition, param);// 应用查询var query = dbContext.Houses.Where(expr);foreach (var house in query){Console.WriteLine(house.Owner);}
    }
    
  2. 生成的 SQL

    SELECT * FROM Houses WHERE Owner like '%tom%'
    

示例2

  1. 动态过滤价格

    using System.Linq.Expressions;// 参数表达式:代表实体对象(如 p => ... 中的 p)
    ParameterExpression param = Expression.Parameter(typeof(House), "p");// 属性访问:p.Price
    MemberExpression priceProperty = Expression.Property(param, "Price");// 常量值:100
    ConstantExpression constant = Expression.Constant(100.0);// 比较表达式:p.Price > 100
    BinaryExpression priceComparison = Expression.GreaterThan(priceProperty, constant);// 组合为 Lambda 表达式
    Expression<Func<House, bool>> expr = Expression.Lambda<Func<House, bool>>(priceComparison, param);// 应用查询
    var query = dbContext.Houses.Where(expr);
    
  2. 生成的SQL

    SELECT * FROM T_Houses WHERE Price > 100
    

示例3

  1. 组合多个表达式(动态查询中,常需要组合多个条件(如 AND/OR))

    public static Expression<Func<House, bool>> BuildDynamicFilter(string paramOwnerStr,string paramPriceStr,
    string nameKeyword, double? minPrice)
    {ParameterExpression param = Expression.Parameter(typeof(House), "p");Expression finalExpr = Expression.Constant(true); // 初始条件:true// 条件1:名称包含关键字if (!string.IsNullOrEmpty(nameKeyword)){MemberExpression nameProperty = Expression.Property(param, paramOwnerStr);MethodInfo containsMethod = typeof(string).GetMethod("Contains", new[] { typeof(string) });ConstantExpression keywordConstant = Expression.Constant(nameKeyword);MethodCallExpression nameCondition = Expression.Call(nameProperty, containsMethod, keywordConstant);finalExpr = Expression.AndAlso(finalExpr, nameCondition);}// 条件2:价格 >= minPriceif (minPrice.HasValue){MemberExpression priceProperty = Expression.Property(param, paramPriceStr);ConstantExpression minPriceConstant = Expression.Constant(minPrice.Value);BinaryExpression priceCondition = Expression.GreaterThanOrEqual(priceProperty, minPriceConstant);finalExpr = Expression.AndAlso(finalExpr, priceCondition);}return Expression.Lambda<Func<House, bool>>(finalExpr, param);
    }
    
     // 使用:using (MyDBContext dbContext =new MyDBContext()){var filter = BuildDynamicFilter("Owner","Price","Tom", 2000.0);var query = dbContext.Houses.Where(filter);foreach (var house in query){Console.WriteLine(house.Owner);}}
    
  2. 生成的SQL

     SELECT [t].[Id], [t].[Name], [t].[Owner], [t].[Price], [t].[RowVersion]FROM [T_Houses] AS [t]WHERE [t].[Owner] LIKE N'%Tom%' AND [t].[Price] >= 2000.0E0
    

高级技巧:表达式合并

  1. 如果需要组合两个已有的表达式(如 expr1 && expr2),需统一参数。

  2. 示例:合并两个表达式

     public static Expression<Func<T, bool>> CombineAnd<T>(Expression<Func<T, bool>> expr1,Expression<Func<T, bool>> expr2){var param = Expression.Parameter(typeof(T));var body1 = ReplaceParameter(expr1.Body, expr1.Parameters[0], param);var body2 = ReplaceParameter(expr2.Body, expr2.Parameters[0], param);return Expression.Lambda<Func<T, bool>>(Expression.AndAlso(body1, body2),param);}private static Expression ReplaceParameter(Expression expression,ParameterExpression oldParam,ParameterExpression newParam){return new ParameterReplacer(oldParam, newParam).Visit(expression);}class ParameterReplacer : ExpressionVisitor{private readonly ParameterExpression _oldParam;private readonly ParameterExpression _newParam;public ParameterReplacer(ParameterExpression oldParam, ParameterExpression newParam){_oldParam = oldParam;_newParam = newParam;}protected override Expression VisitParameter(ParameterExpression node){return node == _oldParam ? _newParam : node;}}
    
     使用:
    Expression<Func<House, bool>> expr1 = p => p.Owner.Contains("Tom");
    Expression<Func<House, bool>> expr2 = p => p.Price > 2000;
    var combinedExpr = CombineAnd(expr1, expr2);
    var query2 = dbContext.Houses.Where(combinedExpr);
    foreach (var house in query2)
    {Console.WriteLine(house.Owner);
    }
    
  3. 生成的SQL

    SELECT [t].[Id], [t].[Name], [t].[Owner], [t].[Price], [t].[RowVersion]FROM [T_Houses] AS [t]WHERE [t].[Owner] LIKE N'%Tom%' AND [t].[Price] > 2000.0E0
    

三、ExpressionTreeToString

ExpressionTreeToString 是一个第三方库,用于将 LINQ 表达式树(Expression)转换为可读的字符串形式,帮助开发者调试和分析表达式树的结构。
输出的所有代码都是对于工厂方法的调用,且调用工厂方法的时候都省略了Expression类,手动添加Expression或者using static System.Linq.Expressions.Expression;

安装方法

  1. 通过 NuGet 包管理器安装

    Install-Package ExpressionTreeToString
    

基本用法

  1. 示例

    Expression<Func<House, bool>> exp1 = b => b.Owner.Contains("tom");
    Expression<Func<House, bool>> exp2 = b => b.Price > 2000;
    //转换为字符串(支持多种格式化选项)
    //string exprString = expr.ToString("C#", "Dynamic LINQ");//Console.WriteLine(exp1.ToString("Factory methods", "C#"));
    Console.WriteLine(exp2.ToString("Factory methods", "C#"));
    
    //输出结果展示
    // using static System.Linq.Expressions.Expressionvar b = Parameter(typeof(House),"b"
    );Lambda(GreaterThan(MakeMemberAccess(b,typeof(House).GetProperty("Price")),Constant(2000)),b
    )
    

支持的格式化风格

ExpressionTreeToString 提供多种输出格式,方便不同场景使用:

  1. C# 语法风格:ToString(“C#”)
    接近 C# 代码的直观表示。
  2. Visual Basic 语法风格:ToString(“VB”)
    类似 VB 语法。
  3. 表达式树结构:ToString(“Object notation”)
    显示表达式树的节点结构(如 BinaryExpressionParameterExpression)。
  4. 调试视图:ToString(“DebugView”)
    Visual Studio 调试器中表达式树的显示一致。

四、注意事项

  1. 不支持所有 C# 方法:某些方法(如 ToString())无法转换为 SQL,会导致运行时错误。
  2. 调试技巧:通过 query.ToQueryString() 查看生成的 SQL
  3. 性能:表达式树构建在内存中完成,复杂逻辑可能影响启动性能。

总结

通过灵活使用表达式树,可以极大增强 EF Core 查询的灵活性,同时保持高效的服务器端执行。


http://www.ppmy.cn/embedded/178218.html

相关文章

力扣经典算法篇-4-删除有序数组中的重复项 II(中等)

题干&#xff1a; 给你一个有序数组 nums &#xff0c;请你 原地 删除重复出现的元素&#xff0c;使得出现次数超过两次的元素只出现两次 &#xff0c;返回删除后数组的新长度。 不要使用额外的数组空间&#xff0c;你必须在 原地 修改输入数组 并在使用 O(1) 额外空间的条件下…

【Bug】记录2025年遇到的Bug以及修复方案

--------------------------------------------------------分割线 2025.3.25-------------------------------------------------------windows环境下通过命令行终端&#xff08;必须是命令行下&#xff0c;直接赋值传递&#xff0c;代码正常&#xff09;的形式传递字符串时&a…

uv vs pip 速度实测

前文 使用 uv 管理 Python 项目 介绍了 uv 的主要使用过程. 相较于传统的 pip 不仅是功能更丰富, 速度也是嘎嘎快. 本文就来做一个实际速度测试对比. 以下测试均使用清华大学 pypi 镜像, 在 Docker 环境中分别启动新的容器进行无本地缓存的冷安装. 使用 Docker 镜像 public.ecr…

【工具变量】上市公司供应链稳定性数据两个维度(2013-2023年)

供应链稳定性是指供应链在面对各种内外部因素的冲击和不确定性时&#xff0c;能够保持持续、顺畅运作的能力&#xff0c;而供应链稳定性指数是用于评估企业在其供应链管理中保持稳定性的一个重要指标。本分享数据参考钟涛&#xff08;2022&#xff09;、董浩和闫晴&#xff08;…

`git commit --amend` 详解:修改提交记录的正确方式

文章目录 git commit --amend 详解&#xff1a;修改提交记录的正确方式1. 修改提交信息2. 补充遗漏的文件3. 结合 --amend 进行交互式修改4. 已推送提交的修改总结 git commit --amend 详解&#xff1a;修改提交记录的正确方式 git commit --amend 用于修改最近一次的提交&…

基于 GEE 的研究区 1986-2024 年年均归一化植被指数 NDVI 时间序列分析

目录 1 代码解析 1.1 初始化与地图设置 1.2 数据预处理函数 1.3 云去除函数 1.4 NDVI计算函数 1.5 数据集加载与处理 1.6 年均NDVI计算与导出 1.7 时间序列影像集合 1.8 绘制时间序列图表 2 完整代码 3 运行结果 1 代码解析 1.1 初始化与地图设置 var roi = table…

日程公布| 第八届地球空间大数据与云计算前沿大会与集中学习(3号通知)

日程公布| 第八届地球空间大数据与云计算前沿大会与集中学习&#xff08;3号通知&#xff09; 日程公布| 第八届地球空间大数据与云计算前沿大会与集中学习&#xff08;3号通知&#xff09;

C++学习笔记(二十九)——list

一、std::list &#xff08;1&#xff09;list与其适用场景 std::list 是 C的STL&#xff08;标准模板库&#xff09;中的双向链表容器&#xff0c;支持高效的插入、删除操作&#xff0c;适用于频繁在容器中间插入或删除元素的场景。 特点&#xff1a; 双向链表&#xff08…