LINQ,查询大过滤器

时间:2009-01-29 17:57:34

标签: c# asp.net-mvc linq

我正在构建一个应用程序的一部分,该部分围绕从数据库中提取有关事务的信息。由于数据的性质,表中有许多我想要过滤的列。我有一个包含15个字段的过滤器选择框,我希望能够为LINQ语句构建一个where子句。当我想要某些字段为空时,有趣的部分就出现了。例如,我希望能够过滤以下任何一个或全部:

  • 交易类型
  • 响应代码
  • 交易金额
  • 更多

我可以建立一个看起来像

的谓词
Func<Transaction, bool> pred = t => t.ResponseCode == ResponseCode && t.TransactionType == TransactionType && t.TransactionAmount > 100.00;

但是为了能够选择在谓词中包含哪些字段我concatenating the predicates

Func<Transaction, bool> pred = t => true;
if(ResponseCode != null)
   pred.AndAlso(t => t.ResponseCode == ResponseCode);
// Rinse and repeat

然后将该谓词传递给LINQ语句的where子句。

这完全按我想要的方式工作,但相当复杂。有没有其他方法可以做到这一点?

更新: 感谢正义的评论。我没有使用LINQ to SQL,我在存储库中的对象集合上使用LINQ。您将如何以编程方式构建表达式过滤器?

2 个答案:

答案 0 :(得分:8)

  • 在动态SQL中...由于您只有一个WHERE子句 - 您必须使用AND连接谓词。
  • 在linq查询构造中...您可以获得任意数量的WHERE子句。 Linq会在翻译查询时将它们和你们联系在一起。

示例:

IQueryable<Transaction> query = db.Transactions;

if (filterByTransactionType)
{
  query = query.Where(t => t.TransactionType == theTransactionType);
}
if (filterByResponseCode)
{
  query = query.Where(t => t.ResponseCode == theResponseCode);
}
if (filterByAmount)
{
  query = query.Where(t => t.TransactionAmount > theAmount);
}

另一个例子:

List<Expression<Func<Transaction, bool>>> filters = GetFilterExpressions();

IQueryable<Transaction> query = db.Transactions;
filters.ForEach(f => query = query.Where(f));

答案 1 :(得分:0)

首先,您需要将Expression<Func<Transaction, bool>>用于LINQ-to-SQL(这是您尝试使用的内容,它与LINQ不同)。

其次,您可以使用Expression<Func<Transaction, bool>>命名空间以编程方式构建System.Linq.Expression

您将无法使用LINQ本身使用以编程方式构建的表达式查询数据库。您需要使用查询扩展方法,而不是使用查询运算符:例如,您需要使用from p in db.People where p.Age > 50 select p.Name而不是db.People.Where(p => p.Age > 50)。您可以使用此样式添加过滤器:db.People.Where(myFilter),其中myFilter = new Expression<Func<Person, bool>>(p => p.Age > 50)。在您的情况下,myFilter将是您的以编程方式构建的过滤器,而不是使用lambda表达式语法创建的过滤器。