我正在尝试构建一个 nHibernate 条件表达式来执行搜索。
给定以下数据模型:
- 一个操作可能有零个或多个 session 。
- 一个操作可能是零 更多操作类型。
我想根据以下条件搜索所有 session :
- (必填)在哪里操作 IsActive 标志为真,IsPublished 标志为真
o(可选)并且操作状态/结束日期在用户指定的日期范围内
o(可选)并且 session 区域 ID 与用户指定的 ID 匹配
o(可选)并且 session 划分与用户指定的 id 匹配
o(可选)以及 Operation.OperationTypes 在用户指定的类型 ID 列表中的位置
我会在 sql 中将其表达为(假定已提供所有可选参数):
SELECT
[Session].*
FROM
[OperationTypeOperation]
LEFT OUTER JOIN
[Operation] ON [OperationTypeOperation].[OperationId] = [Operation].[OperationId]
RIGHT OUTER JOIN
[Session] ON [Operation].[OperationId] = [Session].[OperationId]
WHERE
([Operation].[IsPublished] = 1)
AND
([Operation].[IsActive] = 1)
AND
([Session].[RegionId] = 66)
AND
([Session].[DivisionId] = 99)
AND
([Operation].[AdvertisingStartDate] < GETDATE())
AND
([Operation].[AdvertisingEndDate] > GETDATE())
AND
([OperationTypeOperation].[OperationTypeId] IN (1, 2, 3))
在我的 nHibernate 查询中:
public PagedResult<Session> Search(int? regionId, int? divisionId, DateTime? startDate, DateTime? endDate, IList<int> operationTypeId, int itemsPerPage, int page)
{
var criteria = _session.CreateCriteria(typeof(Session))
.Add(Expression.Eq("IsActive", true))
.Add(Expression.Eq("AcceptingApplications", true))
.AddOrder(new Order("StartDate", false))
;
if (regionId.HasValue)
criteria.Add(Expression.Eq("Region.Id", regionId.Value));
if (divisionId.HasValue)
criteria.Add(Expression.Eq("Division.Id", divisionId.Value));
if (startDate.HasValue)
criteria.Add(Expression.Ge("StartDate", startDate.Value));
if (endDate.HasValue)
criteria.Add(Expression.Le("EndDate", endDate.Value));
//Add the operation types
if (operationTypeId.Count > 0)
{
var operationTypes = new Collection<OperationType>();
foreach (int id in operationTypeId)
{
operationTypes.Add(_session.Get<OperationType>(id));
}
//Join on the operations
criteria.CreateCriteria("Operation")
.Add(Expression.Eq("IsPublished", true))
.Add(Expression.Eq("IsActive", true))
.Add(Expression.Le("AdvertisingStartDate", DateTime.Now))
.Add(Expression.Ge("AdvertisingEndDate", DateTime.Now))
.CreateAlias("OperationTypes", "operationTypes", JoinType.InnerJoin)
.Add(Expression.In("OperationTypes", operationTypes))
.SetResultTransformer(new DistinctRootEntityResultTransformer())
;
}
else
{
//Join on the operations
criteria.CreateCriteria("Operation")
.Add(Expression.Eq("IsPublished", true))
.Add(Expression.Eq("IsActive", true))
.Add(Expression.Le("AdvertisingStartDate", DateTime.Now))
.Add(Expression.Ge("AdvertisingEndDate", DateTime.Now))
;
}
return criteria.ToPagedResult<Session>(itemsPerPage, page);
}
我的 nHibernate 函数提示操作类型并抛出异常“无法将集合与 InExpression 一起使用”。另外,我不确定我是否正确过滤了连接表。任何人都可以权衡将上述 sql 编写为 nHibernate 表达式的正确方法吗?
最佳答案
我假设 OperationType
是一个实体类(不是枚举)。您不能使用 In
带有实体列表。您可以使用 ID 加入。
criteria
.CreateCriteria("Operation")
// add other expressions
.CreateCriteria("OperationTypes", "operationTypes", JoinType.LeftOuterJoin)
.Add(Expression.In("operationTypes.Id", operationTypeId))
我假设 _session.Get(id) 执行数据库查询。无论如何你应该避免这种情况。
PS:如果 OperationType 被映射为复合集合(带有 <composite-element>
标记),那么不幸的是,您无法使用条件加入它。我不确定是否有解决方法。但是有一个补丁正在等待中。
关于c# - 在 nHibernate 中过滤连接,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/600865/