0

这类似于这个问题:NHibernate - Implement "NOT IN" query using ICriteria

但这对我的解决方案不起作用。

我有一个订单:

class Order
{
   DateTime CreatedDate { get; set;} 
   Person CreatedBy { get; set; }
}

和一个人:

class Person
{ 
   string Name { get; set;} // etc.
}

我想检索所有在某个日期范围内没有订单的人。SQL 最终会是这样的:

SELECT *
FROM Person
WHERE Id NOT IN (SELECT PersonId
                 FROM Order
                 WHERE CreatedDate BETWEEN '2012-01-01' and '2012-01-31')

当我使用上述问题中提供的解决方案进行查询时,如果有 2100 人或更多人(SQL 不允许那么多参数),则它会失败。

此外,我无法将 Orders 集合添加到人员,因为这将涉及提取太多数据(我只想要一个日期范围)

我怎样才能用 NHibernate 做到这一点?

4

2 回答 2

3

您可以使用子查询...

var peopleWithOrdersInRange = QueryOver.Of<Order>()
    .WhereRestrictionOn(x => x.CreatedDate).IsBetween(fromDate).And(toDate)
    .SelectGroup(x => x.CreatedBy.Id);

var results = Session.QueryOver<Person>()
    .WithSubquery.WhereProperty(x => x.Id).NotIn(peopleWithOrdersInRange)
    .List();

这将产生确切的 SQL(在子查询中添加按人员 ID 分组)

于 2012-08-08T18:12:30.973 回答
2

当我使用上述问题中提供的解决方案进行查询时,如果有 2100 人或更多人(SQL 不允许那么多参数),则它会失败。

据我了解,您确实在将订单集合传递给查询之前将其具体化。

这是一个应该可以工作的 linq 查询

var unwantedPersons = 
   from order in session.Query<Order>()
   where order.CreatedDate >= startDate && 
         order.CreatedDate <= endDate
   select order.CreatedBy.Id // NOTE: do not add .ToList() or .ToArray() here.
                             // It *should be* IQueryable<Person> 

var personsWitoutOrders = 
    (from person in session.Query<Person>()
     where !unwantedPersons.Contains(person.Id)
     select person).ToArray();
于 2012-08-08T18:04:03.710 回答