在使用Linq查询的时候,特别是如果你在使用Entiry Framwork,有时会遇到动态查询的情况(客户的查询条件是不固定的拼接查询)。
我们能想到的第一方案应该是拼接SQL,的确这样是可以达到我们的目的的。但这样又会破坏程序的一至性,本来使用Entiry Framwork的目标就是用面向对象的方式操纵数据库,这样我们又要开始写SQL语句了。

其实我一开始也是这样做的直到有一天我们部门的美女程序员给我介绍LinqKit,我才开始用PredicateBuilder来拼接Predicate委托。

PredicateBuilder是LinqKit库的一部分,下面是PredicateBuilder源代码:

  1. using System;
  2. using System.Linq;
  3. using System.Linq.Expressions;
  4.  
  5. namespace LinqUtil
  6. {
  7. public static class PredicateBuilder
  8. {
  9. public static Expression<Func<T, bool>> True<T>()
  10. {
  11. return f => true;
  12. }
  13.  
  14. public static Expression<Func<T, bool>> False<T>()
  15. {
  16. return f => false;
  17. }
  18.  
  19. public static Expression<Func<T, bool>> Or<T>(this Expression<Func<T, bool>> expr1,
  20. Expression<Func<T, bool>> expr2)
  21. {
  22. var invokedExpr = Expression.Invoke(expr2, expr1.Parameters.Cast<Expression>());
  23. return Expression.Lambda<Func<T, bool>>
  24. (Expression.OrElse(expr1.Body, invokedExpr), expr1.Parameters);
  25. }
  26.  
  27. public static Expression<Func<T, bool>> And<T>(this Expression<Func<T, bool>> expr1,
  28. Expression<Func<T, bool>> expr2)
  29. {
  30. var invokedExpr = Expression.Invoke(expr2, expr1.Parameters.Cast<Expression>());
  31. return Expression.Lambda<Func<T, bool>>
  32. (Expression.AndAlso(expr1.Body, invokedExpr), expr1.Parameters);
  33. }
  34. }
  35. }

这里主要是介绍PredicateBuilder的使用,如想了解PredicateBuilder具体是怎么实现的可以看看这篇文章:传送门

引入PredicateBuilder依赖

  • 用VS的NuGet安装LinqKit
  • 直接引用上面的源码

创建PredicateBuilder对象

  1. var where = PredicateBuilder.True<TrendStatics>();

可以理解为创建一个初始化为True的Predicate。

注意:如果你是要创建一个OR组成的Predicate就不能把它初始化为True因为这样这个表达试永远为True了。

  1. var where = PredicateBuilder.False<TrendStatics>();

可以理解为创建一个初始化为False的Predicate。

注意:如果你是要创建一个AND组成的Predicate就不能把它初始化为False因为这样这个表达试永远为False了。

PredicateBuilder对象拼接

现在你可以对Predicate进行各种拼接了

  • 全And:
    1. var where = PredicateBuilder.True<int>();
    2. where = where.And(x => x >= );
    3. where = where.And(x => x <= );
    4. var res = list.Where(where.Compile());
  • 全Or:
    1. var list = Enumerable.Range(, );
    2. var where = PredicateBuilder.False<int>();
    3. where = where.Or(x => x == );
    4. where = where.Or(x => x == );
    5. var res = list.Where(where.Compile());
  • 各种组合:
    1. var list = Enumerable.Range(, );
    2. var where = PredicateBuilder.True<int>();
    3. where = where.And(x => x >= );
    4. where = where.And(x => x <= );
    5. var subwhere = PredicateBuilder.False<int>();
    6. subwhere = subwhere.Or(x => x == );
    7. subwhere = subwhere.Or(x => x == );
    8. where = where.And(subwhere);
    9. var res = list.Where(where.Compile());

PredicateBuilder对象使用

  • 针对集合Linq查询

你可以这样用:

  1. var res = list.Where(where.Compile());

你还可以这样用:

  1. var res = list.AsQueryable().Where(where);
  • 针对Entity Framework:
  1. var res = table.Where(where.Expand());

转载请注明出处:http://www.cnblogs.com/keitsi/p/5621136.html

用PredicateBuilder实现Linq动态拼接查询的更多相关文章

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

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

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

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

  3. .NetCore 使用 Linq 动态拼接Expression表达式条件来实现 对EF、EF Core 扩展查询排序操作

    相信在使用EF的时候对查询条件或者排序上的处理令人心烦,下面我们就来动态拼接表达式解决这一问题 当我们在查询中使用Where的时候可以看到如下参数 下面我们就来扩展 Expression<Fun ...

  4. EF动态拼接查询

    1.业务中遇到个问题,需要查询如某表的id为1或者2或者3,这里是根据传递参数获取如:传递1,2或者1,3或者1,2,3这里在sql中很好拼接如下: or id= or name=3//3代表另一个字 ...

  5. Linq to Entity 动态拼接查询条件(重点是OR)

    public static class PredicateExtensions { /// <summary> /// 机关函数应用True时:单个AND有效,多个AND有效:单个OR无效 ...

  6. [C#] Linq 动态条件查询

    应用背景:以货品为例,在基础数据中配置货品的判断规则,要根据这个规则筛选出符合条件的集合. 创建货品类 public class Product { public string Name { get; ...

  7. Linq动态查询与模糊查询 ---转

    Linq动态查询与模糊查询(带源码示例) 继LINQ动态组合查询PredicateExtensions讲解 ----- 在用上面的方法时遇到了些问题 解决 LINQ to Entities 不支持 L ...

  8. Asp.net Core C#进行筛选、过滤、使用PredicateBuilder进行动态拼接lamdba表达式树并用作条件精准查询,模糊查询

    在asp.net core.asp.net 中做where条件过滤筛选的时候写的长而繁琐不利于维护,用PredicateBuilder进行筛选.过滤.LInq配合Ef.core进行动态拼接lamdba ...

  9. linq里lambda写的join查询,并附加动态拼接的条件,条件为enum类型的查询

    因为查询条件不固定的原因,sql式的linq查询没法动态拼接条件. 网上搜的资料整理之后终于解决. 参考资料: enum使用 http://blog.csdn.net/slowlifes/articl ...

随机推荐

  1. brew 更新

    更新: brew update brew update —system 安装, 如:brew install unrar 卸载, 如:brew uninstall unrar

  2. (phpmyadmin error)Login without a password is forbidden by configuration (see AllowNoPassword) in ubuntu

    1.Go to /etc/phpmyadmin/config.inc.php and open it your favorite editor. 2.Search for below line of ...

  3. Visual Studio Profiler 跟踪检查每个exe dll 性能 执行时间 CPU占用情况的方法

  4. delphi请求idhttp数据

    idhttp ss : TStringStream; begin ss := TStringStream.)); { 指定gb2312的中文代码页,或者54936(gb18030)更好些 utf8 对 ...

  5. vim之vba文件

    [vim之vba文件] Vimball 官方描述: The vimball plugin facilitates creating, extracting , and listing the cont ...

  6. C++11对象构造的改良

    [C++11对象构造的改良] C++03中一个构造函数无法构造另一个构造函数,因为A()实际上意味着生成一个临时对象,存在语音混淆.详情请看参考2. C++11中允许直接在初始化列表中调用其它的构造函 ...

  7. HDU 4612 Warm up(2013多校2 1002 双连通分量)

    Warm up Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65535/65535 K (Java/Others)Total Su ...

  8. keil编译时出现*** WARNING L2: REFERENCE MADE TO UNRESOLVED EXTERNAL

    *** WARNING L2: REFERENCE MADE TO UNRESOLVED EXTERNAL *** WARNING L1: UNRESOLVED EXTERNAL SYMBOL 解决: ...

  9. 利用HTML5开发Android(7)---HTML5本地存储之Database Storage

    在上一篇<HTML5本地存储之Web Storage篇>中,简单介绍了如何利用localStorage实现本地存储:实际上,除了sessionStorage和localStorage外,H ...

  10. OC:关于Itunes你了解多少?

    在苹果电脑里有一个 iTunes 简单的说iTunes是苹果公司开发的一个应用程序,由苹果电脑在2001年1月10日于旧金山的 Macworld Expo 推出的一款数字媒体播放应用程序,用于播放以及 ...