将ravendb查询作为Func <t,bool =“”>传递</t,>

时间:2012-06-25 14:13:13

标签: c# ravendb

我正在玩RavenDb并想知道我是否遗漏了一些明显的东西。

事情是,如果我传递这样的查询:

  var name = "test";
  posts = RavenSession.Query<Post>()
         .Where(x => x.Tags.Any(y => y == name))
         .OrderByDescending(x => x.CreatedAt)
         .Take(5);

它工作正常,如果我使用Func<T, bool>编写等效的(IMO),它不会崩溃,但在条件中缺少查询:

 var name = "test";     
 Func<Post, bool> selector = x => x.Tags.Any(y => y == name);
 posts = RavenSession.Query<Post>()
         .Where(x => selector(x))
         .OrderByDescending(x => x.CreatedAt)
         .Take(5);

Profiler输出如下:

  

查询=   开始= 0   的pageSize = 5   聚集=无   排序= -CreatedAt

更新: 如果我使用表达式而不是Func,它可以工作,所以我想我可能记得Func和Linq有些错误,所以写了一个简单的测试:

var range = Enumerable.Range(1, 50);

Func<int, bool> selector = x => x == 42;
var filtered = range.Where(x => selector(x));

所以现在只是为什么Raven Db查询构建器的行为不同。

2 个答案:

答案 0 :(得分:9)

尝试改为使用Expression

Expression<Func<Post, bool>> selector = x => x.Tags.Any(y => y == name);

Where(x => selector(x))更改为Where(selector)

需要Expression,因为RavenDb可以从中构建表达式树,这允许它将逻辑转换为数据库查询。它无法从Func<Post, bool>构建表达式树,因此它可能会忽略它,抛出一个异常或RavenDb指定的创建者。

答案 1 :(得分:3)

作为对您的更新的回复,Func<>上的IEnumerable<>Expression<Func<>> IQueryable<>上看起来语法完全相同的行为存在显着差异。这不只是Raven,而是任何IQueryable<>源,例如LINQ to SQL或Entity Framework。