事情的起因是公司一个小伙子问了我个问题 “海哥,来帮我看下这段代码怎么不行”

Func<Report,bool> nameFilter = x=>x.Name == "test";
DbContext.Report.Where(x=>x.State==1 && nameFilter(x));

我一看,好家伙,这么骚的代码都能让你想出来,正常情况下用Linq To Object是可以这么操作的,但是EF的IQueryable查询是不能这么操作的。

Linq To Object是直接执行表达式,他就是个委托方法,里面嵌套多少层委托和方法都是能直接执行的

IQueryable并不会执行表达式和方法,是把表达式转换为对应的Sql语句来执行,解析到nameFilter的时候他就懵逼了,这是啥玩意儿啊,sql里面没有这种东西啊,他就转换不了了。

小伙子知道后明细很失望,那不能啊,也不是我想显摆我的技术,就是想让小伙子能继续他的骚操作,给他来点海克斯科技与狠活。

解决方案:

//表达式
Func<Report,bool> nameFilter = x=>x.Name == "test";
Func<Report,bool> stateFilter = x=>x.State==1;
//合并为
Func<Report,bool> whereFilter = x=>x.Name == "test" && x.State==1; //调用
DbContext.Report.Where(whereFilter);

完美解决

那怎么合并,当然得自己构造一个新的表达式,构造表达式需要用到Expression类,如果没有用过这个类,可以按照下面的方式来调试看看一个表达式转换为表达式树是怎么样的。


TestExpression(x=>x.Name == "test",x=>x.State==1); public static void TestExpression(Expression<Func<Report, bool>> left,Expression<Func<Report, bool>> right)
{
//调试查看expression对象
var bodyLeft = left.Body;//这个就是x.Name == "test"
var bodyRight = right.Body;//这个就是x.State==1
}

好,这里我们能获取到表达式的Body,然后使用Expression类能很好的合并两个表达式的body

var andAlso = Expression.AndAlso(bodyLeft ,bodyRight);//x.Name == "test" && x.State==1

这样还不行,这两个表达式是两个不同的委托对象,他们的参数x也是两个不同的对象,合并了又没完全合并

这就需要用到ExpressionVisitor类来递归表达式树,把两个表达式的参数替换为同一个参数。

    /// <summary>
/// 替换表达式参数
/// </summary>
public class ReplaceExpressionVisitor : ExpressionVisitor
{
private Expression _leftParameter; public ReplaceExpressionVisitor(Expression leftParameter)
{
_leftParameter= leftParameter;
} protected override Expression VisitParameter(ParameterExpression node)
{
return _leftParameter;
}
}

最终


TestExpression(x=>x.Name == "test",x=>x.State==1); public static void TestExpression(Expression<Func<Report, bool>> left,Expression<Func<Report, bool>> right)
{
//调试查看expression对象
var bodyLeft = left.Body;//这个就是x.Name == "test"
var bodyRight = right.Body;//这个就是x.State==1
var leftParameter = left.Parameters[0];
//表达式递归访问
var visitor =new ReplaceExpressionVisitor(leftParameter);
//替换参数
bodyRight = visitor.Visit(bodyRight);
//合并表达式
var expression = Expression.AndAlso(bodyLeft , bodyRight);
//构建表达式
var whereExpression= Expression.Lambda<Func<Report, bool>>(expression , left.Parameters);
//编译表达式
var whereFilter = whereExpression.Compile();
//使用
DbContext.Report.Where(whereFilter);
}

正想给小老弟显摆一下的时候,他又去写其他骚代码了

骚不过骚不过,完善一下列子,下面是完整的代码

小嫩手不想动的小伙伴可以直接nuget上查找DynamicExpression.Core,直接使用

更多源码看本人github

    /// <summary>
/// 替换表达式参数
/// </summary>
public class ReplaceExpressionVisitor : ExpressionVisitor
{
private Dictionary<Expression, Expression> _parameters; public ReplaceExpressionVisitor(Dictionary<Expression,Expression> parameters)
{
_parameters = parameters;
} protected override Expression VisitParameter(ParameterExpression node)
{
if (_parameters.TryGetValue(node, out Expression _newValue))
{
return _newValue;
}
return base.Visit(node);
}
}
    /// <summary>
/// 表达式扩展
/// </summary>
public static class ExpressionExtension
{ /// <summary>
/// 使用AndAlso合并表达式
/// </summary>
/// <param name="exprs"></param>
/// <returns></returns>
public static Expression<T> AndAlso<T>(this IList<Expression<T>> exprs)
{
if (exprs.Count == 0) return null;
if (exprs.Count == 1) return exprs[0]; var leftExpr = exprs[0];
var left = leftExpr.Body;
for (int i = 1; i < exprs.Count; i++)
{
var expr = exprs[i];
var visitor = GetReplaceExpressionVisitor(expr.Parameters, leftExpr.Parameters);
var right = visitor.Visit(expr.Body);
left = Expression.AndAlso(left, right);
}
return Expression.Lambda<T>(left, leftExpr.Parameters);
} /// <summary>
/// 使用AndAlso合并表达式
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="left"></param>
/// <param name="right"></param>
/// <returns>left AndAlso right</returns>
public static Expression<T> AndAlso<T>(this Expression<T> left, Expression<T> right)
{
return AndAlso(new List<Expression<T>>() { left, right });
} /// <summary>
/// 使用OrElse合并表达式
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="exprs"></param>
/// <returns></returns>
public static Expression<T> OrElse<T>(this IList<Expression<T>> exprs)
{
if (exprs.Count == 0) return null;
if (exprs.Count == 1) return exprs[0]; var leftExpr = exprs[0];
var left = leftExpr.Body;
for (int i = 1; i < exprs.Count; i++)
{
var expr = exprs[i];
var visitor = GetReplaceExpressionVisitor(expr.Parameters, leftExpr.Parameters);
var right = visitor.Visit(expr.Body);
left = Expression.OrElse(left, right);
}
return Expression.Lambda<T>(left, leftExpr.Parameters);
} /// <summary>
/// 使用OrElse合并表达式
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="left"></param>
/// <param name="right"></param>
/// <returns>left OrElse right</returns>
public static Expression<T> OrElse<T>(this Expression<T> left, Expression<T> right)
{
return OrElse(new List<Expression<T>>() { left, right });
} /// <summary>
/// 构建visitor
/// </summary>
/// <param name="oldParameters"></param>
/// <param name="newParameters"></param>
/// <returns></returns>
private static ReplaceExpressionVisitor GetReplaceExpressionVisitor(ReadOnlyCollection<ParameterExpression> oldParameters, ReadOnlyCollection<ParameterExpression> newParameters)
{
Dictionary<Expression, Expression> dic = new Dictionary<Expression, Expression>();
for (int i = 0; i < oldParameters.Count; i++)
{
dic.Add(oldParameters[i],newParameters[i]);
}
return new ReplaceExpressionVisitor(dic);
}
}

使用

string connectString = "Data Source=.;Initial Catalog=RportTest;Integrated Security=True";
var optionsBuilder = new DbContextOptionsBuilder<TestContext>();
optionsBuilder.UseSqlServer(connectString);
using (TestContext ctx = new TestContext(optionsBuilder.Options))
{ Expression<Func<ReportData, bool>> epxr1 = report => report.ID == 2023;
Expression<Func<ReportData, bool>> epxr2 = report => report.Name == "test1"; var epxr3 = new List<Expression<Func<ReportData, bool>>>() { epxr1, epxr2 }; var andPredicate = epxr3.AndAlso();
var andQuery = ctx.ReportData.Where(andPredicate);
string andSql = andQuery.ToQueryString();
var andResult = andQuery.ToList(); var orPredicate = epxr3.OrElse();
var orQuery = ctx.ReportData.Where(orPredicate);
string orSql = orQuery.ToQueryString();
var orResult = orQuery.ToList();
}

最新文章

  1. mysql时间属性之时间戳和datetime之间的转换
  2. Java类额应用
  3. PCIe 32GT/s 含义
  4. WCF分布式开发必备知识(2):.Net Remoting
  5. this和super
  6. 1-5Tomcat 目录结构 和 web项目目录结构
  7. unity, ugui input field
  8. GoAhead 嵌入式web
  9. Windows7配置GPU和Theano编程环境
  10. WinForm 根据屏幕分辨率自适应
  11. Kafka 之 async producer (1)
  12. 黄聪:Microsoft Enterprise Library 5.0 系列教程(一) : Caching Application Block (初级)
  13. SqlServer 汉字转换拼音首字母函数
  14. poj2386
  15. Js特殊字符转义之htmlEscape()方法
  16. 执行PHP脚本时遇到 mysql_connect(): Headers and client library minor version mismatch的解决方法
  17. Python进阶内容(六)--- 函数式编程
  18. Linux - 微软无线鼠标滚动过快问题
  19. slf4j 和 log4j的关系及合用Maven配置
  20. 【转】CentOS系统操作下安装相关各种软件

热门文章

  1. 运维实践-最新Nginx二进制构建编译lua-nginx-module动态链接Lua脚本访问Redis数据库读取静态资源隐式展现
  2. 大数据Hadoop入门教程 | (二)Linux
  3. ABC 203 F - Weed (DP)
  4. 学习ASP.NET Core Blazor编程系列一——综述
  5. EPIC限免提示
  6. HC32L110(五) Ubuntu20.04 VSCode的Debug环境配置
  7. KingbaseES 数据库连接断开问题排查思路
  8. 2021年3月-第02阶段-前端基础-Flex 伸缩布局-移动WEB开发_flex布局
  9. 输入法词库解析(五)极点码表.mb
  10. AVL tree 高度上下界推导