如何在没有建立关系的情况下实现NOT IN

时间:2012-08-08 14:17:01

标签: c# nhibernate icriteria

这与此问题类似: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将不允许这么多参数)。

此外,我无法向人员添加订单集合,因为这会涉及拉方式太多数据(我只想要一个日期范围)。

我怎么能用NHibernate做到这一点?

2 个答案:

答案 0 :(得分: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(在子查询中添加group by people id)

答案 1 :(得分: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();