使用ExpressionVisitor进行lambadaExpression的动态拼接
现有如下实体
public class User
{
public int Id { get; set; }
public string Name { get; set; }
}
根据这个实体创建一个Iqueryable<User>对象queryList
List<User> dList = new List<User> {
new User{Id=,Name="zhansan" },
new User{Id=,Name="lisi" },
new User{Id=,Name="wangwu" },
new User{Id=,Name="zhaoliu" },
new User{Id=,Name="sunqi" },
};
dList的where方法需要一个Func<User, bool> whereFunc参数来进行数据过滤,但很多时候whereFunc是动态的,我的查询条件有时是u=>u.Id==1||u.Id==2,有时条件为u=>u.Id==1||u.Id==2&&u.Name=="lisi"
如此看来条件是不固定的并且属性组合方式多种多样,这样就迫切地需要一种方法可以动态地拼接表达式
如下两个表达式,将演示如何和为一个表达式
Expression<Func<User, bool>> ef = u => u.Id == || u.Id == ;
Expression<Func<User, bool>> ef2 = y => y.Name == "lisi";
首先创建类MyVisitor
public class MyVisitor : ExpressionVisitor
{
Dictionary<ParameterExpression, ParameterExpression> map;
public MyVisitor(Dictionary<ParameterExpression, ParameterExpression> map)
{
this.map = map;
}
protected override Expression VisitParameter(ParameterExpression node)
{
return map[node];
}
}
此类继承自ExpressionVisitor类,此类是expression表达式的工具类,可对expression表达式内部的成员进行替换
首先我自定义了构造函数传入一个字典对象。作用在后面解释
然后我重写了VisitorParameter方法,此方法的参数node即为要修改的expression的参数部分,你的expression有多少个参数就会顺序触发此方法并传入相应参数
准备工作到此就结束了,接下来是拼接过程
var pl = ef.Parameters;
var dic = ef.Parameters.Select((f, i) => new { f, f2 = ef2.Parameters[i] }).ToDictionary(p => p.f2, p => p.f);
MyVisitor myVisitor = new MyVisitor(dic);
var body = myVisitor.Visit(ef2.Body);
var finalBody = Expression.And(ef.Body, body);
var fu = Expression.Lambda<Func<User, bool>>(finalBody, pl).Compile();
var re= dList.Where(fu).ToList();
首先得到字典dic其key为ef2的参数列表,值为ef的参数列表
然后创建刚才自定义的MyVisitor类的对象,并传入字典dic
之后调用myVisitor的Visit方法传入ef2的Body部分,此时根据MyVisitor中对VisitParameter的重写可知
protected override Expression VisitParameter(ParameterExpression node)
{
return map[node];
}
在内部myVisitor对传入的ef2.Body的每一个参数查询字典,并返回值,此值即为ef的参数值,此处就将ef2的参数都替换为ef1的参数了
接下来就是两个表达式的拼接部分
var body = myVisitor.Visit(ef2.Body);
var finalBody = Expression.And(ef.Body, body);
var fu = Expression.Lambda<Func<User, bool>>(finalBody, pl).Compile();
将两个表达式的body部分用and连接,并且使用pl为参数(这很重要,前面做这么多就是为了把ef2的body部分的参数更换为ef的参数)
这样就得到了拼接后的func,传入where作为条件进行查询
var re= dList.Where(fu).ToList();
使用ExpressionVisitor进行lambadaExpression的动态拼接的更多相关文章
- Lambda表达式动态拼接(备忘)
EntityFramework动态组合Lambda表达式作为数据筛选条件,代替拼接SQL语句 分类: C# Lambda/Linq Entity Framework 2013-05-24 06:58 ...
- EF 拉姆达 动态拼接查询语句
EF 动态拼接查询语句 using System; using System.Collections.Generic; using System.IO; using System.Linq; usin ...
- 初识Mybatis框架,实现增删改查等操作(动态拼接和动态修改)
此第一次接触Mybatis框架确实是有点不适应,特别是刚从Hibernate框架转转型过来,那么为什么要使用Mybatis框架,Mybatis框架和Hibernate框架又有什么异同呢? 这个问题在我 ...
- mybatis 使用记录(二) 动态拼接查询条件
2016-12-16 阅读项目代码时,在项目的xml文件中发现如下写法: SELECT student_user_id FROM tbr_student_class WHERE 1=1 <if ...
- 动态拼接 sql的时候 里面 如果有变量的话 按上面的方式进行处理
set @Sql_Sql = N' select top 1 @m_zw=zw,@m_zh=temp from ket where zd=''ddd'' ' print @Sql_Sql EXEC s ...
- 分享动态拼接Expression表达式组件及原理
前言 LINQ大家都知道,用起来也还不错,但有一个问题,当你用Linq进行搜索的时候,你是这样写的 var query = from user in db.Set<User>() ...
- 表达式树动态拼接lambda
动态拼接lambda表达式树 前言 最近在优化同事写的代码(我们的框架用的是dapperLambda),其中有一个这样很普通的场景——界面上提供了一些查询条件框供用户来进行过滤数据.由于dappe ...
- 动态拼接lambda表达式树
前言 最近在优化同事写的代码(我们的框架用的是dapperLambda),其中有一个这样很普通的场景——界面上提供了一些查询条件框供用户来进行过滤数据.由于dapperLambda按条件查询时是传入表 ...
- vue获得当前页面URL动态拼接URL复制邀请链接方法
vue获得当前页面URL动态拼接URL复制邀请链接方法 当前页面完整url可以用 location.href路由路径可以用 this.$route.path路由路径参数 this.$route.par ...
随机推荐
- 禁用自动映射的 Exchange 邮箱
客户最近询问他们无法从用户的Outlook配置文件中删除邮箱.在这种情况下,它们是共享邮箱,并出现在Outlook的左窗格中.但原因和解决方法同样适用于用户邮箱.并且 无论用户邮箱在本地 Exchan ...
- 【PHP】制作日历
本期本博主将讲述两种利用PHP制作日历的方法,由于PHP日期函数的便捷性,使得我们制作日历这一过程变得相当简单 问题描述: 1.取到当前日期,并着色显示:2.根据当前日期,判断本月有多少天,一号是周几 ...
- Unity-奥义技能背景变黑效果
[旧博客转移 - 2016年8月29日 12:51 ] 前段时间做了一个放技能的时候,背景缓慢变黑,放完后再变回来的效果,可以很好的突出技能特效的感觉. 算是一种屏幕后期特效,这个特效说难不难,说简单 ...
- Jsp页面用table表格来让文字和文本框对齐
1, border="0" cellpadding="0" cellspacing="20", 第一个是页边距要设置0,第二个是用来指定单元 ...
- [Oracle]高水位标记(HWM)
(一)高水位标记(High Water Mark,HWM)的概念 所谓高水位标记,是指一个已经分配的段中,已经使用的空间与未使用的空间的分界线.在表的使用过程中,随着数据的不断增多(insert),H ...
- ASP.NET Core 开源论坛项目 NETCoreBBS
ASP.NET Core 轻量化开源论坛项目,ASP.NET Core Light forum NETCoreBBS 采用 ASP.NET Core + EF Core Sqlite + Bootst ...
- ES6 新增命令
let 用来声明变量.它的用法类似于var,但是所声明的变量, 只在let命令所在的代码块内有效. 例: {var a=10; let b=20;}; console.lo ...
- JDK动态代理源码学习
继上一篇博客设计模式之代理模式学习之后http://blog.csdn.net/u014427391/article/details/75115928,本博客介绍JDK动态代理的实现原理,学习一下JD ...
- JS操作数组常用的方法
JS操作Array对象的方法 concat(arr1,arr2,...):连接数组indexOf(value):返回数组中value的第一个索引join(separator):将数组中所有的元素连接由 ...
- sqlserver的触发器练习实例
触发器的概念:它是由事件驱动的,就像java中的监听,当某个事件发生了,就会做一些工作. 下面直接上干货,创建insert触发器.delete触发器.DDL触发器和如何查看触发器定义 1.创建三个表学 ...