这与此问题类似: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做到这一点?
答案 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();