现有如下实体

    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的动态拼接的更多相关文章

  1. Lambda表达式动态拼接(备忘)

    EntityFramework动态组合Lambda表达式作为数据筛选条件,代替拼接SQL语句 分类: C# Lambda/Linq Entity Framework 2013-05-24 06:58 ...

  2. EF 拉姆达 动态拼接查询语句

    EF 动态拼接查询语句 using System; using System.Collections.Generic; using System.IO; using System.Linq; usin ...

  3. 初识Mybatis框架,实现增删改查等操作(动态拼接和动态修改)

    此第一次接触Mybatis框架确实是有点不适应,特别是刚从Hibernate框架转转型过来,那么为什么要使用Mybatis框架,Mybatis框架和Hibernate框架又有什么异同呢? 这个问题在我 ...

  4. mybatis 使用记录(二) 动态拼接查询条件

    2016-12-16 阅读项目代码时,在项目的xml文件中发现如下写法: SELECT student_user_id FROM tbr_student_class WHERE 1=1 <if ...

  5. 动态拼接 sql的时候 里面 如果有变量的话 按上面的方式进行处理

    set @Sql_Sql = N' select top 1 @m_zw=zw,@m_zh=temp from ket where zd=''ddd'' ' print @Sql_Sql EXEC s ...

  6. 分享动态拼接Expression表达式组件及原理

    前言 LINQ大家都知道,用起来也还不错,但有一个问题,当你用Linq进行搜索的时候,你是这样写的 var query = from user in db.Set<User>()      ...

  7. 表达式树动态拼接lambda

    动态拼接lambda表达式树   前言 最近在优化同事写的代码(我们的框架用的是dapperLambda),其中有一个这样很普通的场景——界面上提供了一些查询条件框供用户来进行过滤数据.由于dappe ...

  8. 动态拼接lambda表达式树

    前言 最近在优化同事写的代码(我们的框架用的是dapperLambda),其中有一个这样很普通的场景——界面上提供了一些查询条件框供用户来进行过滤数据.由于dapperLambda按条件查询时是传入表 ...

  9. vue获得当前页面URL动态拼接URL复制邀请链接方法

    vue获得当前页面URL动态拼接URL复制邀请链接方法 当前页面完整url可以用 location.href路由路径可以用 this.$route.path路由路径参数 this.$route.par ...

随机推荐

  1. 如何将txt的多行记录直接导入到mysql数据库

    1.使用工具是navicat for mysql 2.要导入的txt格式要求,第一行为栏位,及个属性名 第二行开始为数据行 如下所示,例如要插入多行账号密码

  2. 邪恶改装2:用单片机实现一次简单的wifi密码欺骗

    0×00 前言 前段时间用TPYBoard v202 做了一个简单的WIFI干扰攻击器(ps :没有看过的小伙伴,可以看一下:http://www.freebuf.com/column/136985. ...

  3. 数据结构之网络流入门(Network Flow)简单小节

    网络流的相关定义: 源点:有n个点,有m条有向边,有一个点很特殊,只出不进,叫做源点. 汇点:另一个点也很特殊,只进不出,叫做汇点. 容量和流量:每条有向边上有两个量,容量和流量,从i到j的容量通常用 ...

  4. ionic项目结构解析

    ionic项目结构解析 原始结构 创建一个IonicDemo项目 'ionic start IonicDemo sidemenu' 这种结构多模块开发比较麻烦,因为view跟controller分开路 ...

  5. Python3中文件处理

    1.txt,xls,doc等文件的使用 f=open("filename","w")   打开一个用于写入的文件,要写入内容时使用f.write("内 ...

  6. Vijos 1034 家族 并查集

    描述 若某个家族人员过于庞大,要判断两个是否是亲戚,确实还很不容易,现在给出某个亲戚关系图,求任意给出的两个人是否具有亲戚关系. 规定:x和y是亲戚,y和z是亲戚,那么x和z也是亲戚.如果x,y是亲戚 ...

  7. unslider插件的使用

    深入理解unslider.js源码 最近用到了一个挺好用的幻灯片插件,叫做unslider.js,就想看看怎么实现幻灯片功能,就看看源码,顺便自己也学习学习.看完之后收获很多,这里和大家分享一下. u ...

  8. Chrome浏览器扩展开发系列之十:桌面通知Notification

    Desktop Notification也称为Web Notification,是在Web页面之外,以弹出桌面对话框的形式通知用户发生了某事件.Web Notification于2015.9.10成为 ...

  9. eclipse 设置 默认编码为 utf-8

    学习javaweb时,开发工具都采用utf-8的编码方式,给eclipse设置默认编码为utf-8的编码方法 菜单 Window -> preference -> General -> ...

  10. 谷歌是如何做代码审查的 | 外刊IT评论 - Google Chrome

    谷歌是如何做代码审查的           本文的作者 Mark CC 在上一篇文章中提到过,我已经不在Google工作了.我还没有想清楚应该去哪里-有两三个非常好的工作机会摆在我面前.因为在这段做决 ...