c# - 在 nHibernate 中过滤连接

标签 c# nhibernate

我正在尝试构建一个 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/

相关文章:

c# - 使用 HttpClient 的随机 401 未授权错误

sql-server - 使用 NHibernate 在同一 ASP.NET 应用程序中支持 Oracle 和 SQL Server 的建议

c# - Moq 和 NHibernate 如何自动创建派生类型?

c# - 在 C# 的属性设置中使用 virtual 和 protected 关键字?

c# - 简易喷油器 : Injecting a property in a base class

c# - Protobuf-net - 仅序列化一些属性

c# - 我如何打开一个新窗口而不是新选项卡或弹出窗口,而是在 c# 中从代码后面单击按钮时打开一个新窗口

c# - 从将部署到 GAC 的自定义 .Net 数据提供程序调用非托管代码时,我应该在哪里部署要使用的 DLL?

c# - 如何从我的域逻辑生成 NHibernate 映射文件和数据库结构?

nhibernate - 在不触发延迟加载或急切加载的情况下将 NHibernate POCO 复制到 DTO