动态组合表达式谓词
假设您要编写实现 SQL 的 LINQ to SQL 或实体框架查询 关键字样式搜索。换句话说,返回其行的查询 描述
包含给定集合的部分或全部 的关键字。
我们可以按以下步骤进行:
IQueryable<Product> SearchProducts (params string[] keywords)
IQueryable<Product> query = dataContext.Products;
foreach (string keyword in keywords)
query = query.Where (p => p.Description.Contains (keyword));
return query;
目前为止,一切都好。但这只处理您想要的情况 以匹配所有指定的关键字。假设 相反,我们想要描述包含任何提供的关键字的产品。我们以前的链接方法 其中 运算符完全没用!我们可以改为连锁联盟运营商,但是 这将是低效的。理想的方法是动态构造一个 执行基于 OR 的 lambda 表达式树 谓语。
在所有会驱使您手动构建的事情中 表达式树,对动态谓词的需求是最常见的 典型的业务应用程序。幸运的是,可以编写一组 简单且可重用的扩展方法从根本上简化了此任务。这是 我们的谓词生成器类的角色。
使用 PredicateBuilder
下面介绍如何使用 PredicateBuilder 解决前面的示例:
IQueryable<Product> SearchProducts (params string[] keywords)
var predicate = PredicateBuilder.False<Product>();
foreach (string keyword in keywords)
predicate = predicate.Or (p => p.Description.Contains (keyword));
return dataContext.Products.Where (predicate);
如果使用实体框架进行查询,请将最后一行更改为:
return objectContext.Products.AsExpandable().Where (predicate);
AsExpandable 方法是 LINQKit 的一部分(见下文)。
试验 PredicateBuilder 的最简单方法是使用 LINQPad。 LINQPad 允许您针对数据库或本地集合即时测试 LINQ 查询,并直接支持 PredicateBuilder(按 F4 并选中“包含谓词生成器”)。
PredicateBuilder 源代码
这是完整的来源:
using System;
using System.Linq;
using System.Linq.Expressions;
using System.Collections.Generic;
public static class PredicateBuilder
public static Expression<Func<T, bool>> True<T> () { return f => true; }
public static Expression<Func<T, bool>> False<T> () { return f => false; }
public static Expression<Func<T, bool>> Or<T> (this Expression<Func<T, bool>> expr1,
Expression<Func<T, bool>> expr2)
var invokedExpr = Expression.Invoke (expr2, expr1.Parameters.Cast<Expression> ());
return Expression.Lambda<Func<T, bool>>
(Expression.OrElse (expr1.Body, invokedExpr), expr1.Parameters);
public static Expression<Func<T, bool>> And<T> (this Expression<Func<T, bool>> expr1,
Expression<Func<T, bool>> expr2)
var invokedExpr = Expression.Invoke (expr2, expr1.Parameters.Cast<Expression> ());
return Expression.Lambda<Func<T, bool>>
(Expression.AndAlso (expr1.Body, invokedExpr), expr1.Parameters);
PredicateBuilder 也作为 LINQKit 的一部分提供,LINQKit 是 LINQ to SQL 和实体框架的生产力工具包。
如果使用 LINQ to SQL,则可以单独使用 PredicateBuilder 源代码。
如果你是 使用实体框架,您将需要完整的 LINQKit - 用于扩展功能。 您可以引用 LINQKit.dll也可以将 LINQKit 的源代码复制到应用程序中。
True 和 False 方法没有什么特别之处:它们只是创建表达式<Func<T,bool>>的方便快捷方式,该表达式最初计算结果为 对或错。所以以下内容:
var predicate = PredicateBuilder.True <Product> ();
只是一个快捷方式:
Expression<Func<Product, bool>> predicate = c => true;
当您通过重复堆叠和/或条件来构建谓词时, 将起点设置为 true 或 false (分别)很有用。如果没有关键字,我们的搜索产品方法仍然有效 提供。
有趣的工作发生在 And 和 Or 方法中。我们首先用第一个表达式调用第二个表达式 表达式的参数。调用表达式调用 另一个使用给定表达式作为参数的 lambda 表达式。我们可以 从第一个表达式的主体和第二个表达式的调用版本创建条件表达式。最后一步是 将其包装在新的 lambda 表达式中。
实体框架的查询处理管道无法 处理调用表达式,这就是需要对查询中的第一个对象调用 AsExpandable 的原因。通过调用 AsExpandable,您可以 激活 LINQKit 的表达式访问者类,用于替换调用 具有实体框架可以理解的更简单构造的表达式。
编写数据访问层的一个有用模式是创建一个 可重用的谓词库。然后,您的查询主要由选择和排序子句组成, 筛选逻辑已扩展到您的库。下面是一个简单的示例:
public partial class Product
public static Expression<Func<Product, bool>> IsSelling()
return p => !p.Discontinued && p.LastSale > DateTime.Now.AddDays (-30);
我们可以通过添加一个使用 PredicateBuilder 的方法来扩展它:
public partial class Product
public static Expression<Func<Product, bool>> ContainsInDescription (
params string[] keywords)
var predicate = PredicateBuilder.False<Product>();
foreach (string keyword in keywords)
predicate = predicate.Or (p => p.Description.Contains (keyword));
return predicate;
这提供了简单性和可重用性的完美平衡, 以及将业务逻辑与表达式管道逻辑分离。检索 描述中包含“黑莓”或“iPhone”的所有产品,以及 正在销售的诺基亚和爱立信,你会这样做:
var newKids = Product.ContainsInDescription ("BlackBerry", "iPhone");
var classics = Product.ContainsInDescription ("Nokia", "Ericsson")
.And (Product.IsSelling());
var query =
from p in Data.Products.Where (newKids.Or (classics))
select p;
粗体中的 And 和 Or 方法解析为 PredicateBuilder 中的扩展方法。
表达式谓词可以通过以下方式执行 SQL 子查询的等效项: 引用关联属性。所以,如果产品有 一个名为“购买”的子实体集,我们可以优化我们的 IsSelling 方法,以仅返回那些 已售出最低单位数量,如下所示:
public static Expression<Func<Product, bool>> IsSelling (int minPurchases)
return prod =>
!prod.Discontinued &&
prod.Purchases.Where (purch => purch.Date > DateTime.Now.AddDays(-30))
.Count() >= minPurchases;
请考虑以下谓词:
p => p.Price > 100 &&
p.Price < 1000 &&
(p.Description.Contains ("foo") || p.Description.Contains ("far"))
假设我们想动态构建它。问题是, 我们如何处理最后一行中两个表达式周围的括号?
答案是先构建括号表达式,然后 然后在外部表达式中使用它,如下所示:
var inner = PredicateBuilder.False<Product>();
inner = inner.Or (p => p.Description.Contains ("foo"));
inner = inner.Or (p => p.Description.Contains ("far"));
var outer = PredicateBuilder.True<Product>();
outer = outer.And (p => p.Price > 100);
outer = outer.And (p => p.Price < 1000);
outer = outer.And (inner);
请注意,对于内部表达式,我们从 谓词生成器。False(因为我们使用的是 Or 运算符)。跟 然而,外部表达式,我们从 PredicateBuilder 开始。True(因为我们使用的是 And 运算符)。
假设数据库中的每个表都有 ValidFrom 和 ValidTo 列,如下所示:
create table PriceList
ID int not null primary key,
Name nvarchar(50) not null,
ValidFrom datetime,
ValidTo datetime
检索自 DateTime.Now 起有效的行(最 常见情况),您将这样做:
from p in PriceLists
where (p.ValidFrom == null || p.ValidFrom <= DateTime.Now) &&
(p.ValidTo == null || p.ValidTo >= DateTime.Now)
select p.Name
当然,粗体字的逻辑很可能会在 多个查询!没问题:让我们在 PriceList 类中定义一个返回可重用表达式的方法:
public static Expression<Func<PriceList, bool>> IsCurrent()
return p => (p.ValidFrom == null || p.ValidFrom <= DateTime.Now) &&
(p.ValidTo == null || p.ValidTo >= DateTime.Now);
好的:我们的查询现在简单多了:
var currentPriceLists = db.PriceLists.Where (PriceList.IsCurrent());
使用PredicateBuilder的And and Or 方法,我们 可以轻松引入其他条件:
var currentPriceLists = db.PriceLists.Where (
PriceList.IsCurrent().And (p => p.Name.StartsWith ("A")));
但是,所有其他同时具有 ValidFrom 和 ValidTo 列的表呢?我们不想重复我们的 IsCurrent 方法 每张桌子!幸运的是,我们可以推广我们的 IsCurrent 方法 泛 型。
第一步是定义接口:
public interface IValidFromTo
DateTime? ValidFrom { get; }
DateTime? ValidTo { get; }
现在我们可以定义一个通用的 IsCurrent 方法,使用 该接口作为约束:
public static Expression<Func<TEntity, bool>> IsCurrent<TEntity>()
where TEntity : IValidFromTo
return e => (e.ValidFrom == null || e.ValidFrom <= DateTime.Now) &&
(e.ValidTo == null || e.ValidTo >= DateTime.Now);
最后一步是在每个类中实现此接口 支持 ValidFrom 和 ValidTo。如果您使用的是 Visual Studio 或 像 SqlMetal 这样的工具来生成你的实体类,在未生成的一半中执行此操作 分部类:
public partial class PriceList : IValidFromTo { }
public partial class Product : IValidFromTo { }
在 LINQPad 中使用 PredicateBuilder
使用 LINQPad,您可以编写和测试查询 比Visual Studio的构建/运行/调试周期快得多。要使用 PredicateBuilder in LINQPad with LINQ to SQL:
按 F4 并选中“包含谓词生成器”
若要在 LINQPad 中使用 PredicateBuilder 和实体框架,请执行以下操作:
按 F4 并添加对 LinqKit 的引用.dll
转自:C# 简而言之 - PredicateBuilder (albahari.com)