3

我是使用 NHibernate 的新手,我很难在网上找到关于如何在不使用 XML 进行映射的情况下为存储过程创建 ClassMap 的清晰示例。我最近使用 Fluent 接口完成了这项工作,并想分享我学到的东西。

有问题的存储过程返回一个像这样的对象:

public class ProductCategoryNavigation
{
    public virtual int CategoryId { get; protected set; }
    public virtual int CategoryNodeId { get; set; }
    public virtual int ParentCategoryNodeId { get; set; }
    public virtual string Name { get; set; }
    public virtual string Title { get; set; }
    public virtual string SeoUrl { get; set; }
    public virtual bool IsActive { get; set; }
    public virtual int DisplayOrder { get; set; }
    public virtual int ProductCount { get; set; }
}

那么,如何创建 NHibernate 用来将存储过程的结果映射到该对象的 ClassMap?

4

2 回答 2

8

ClassMap 看起来像这样:

public sealed class ProductCategoryNavigationMap : ClassMap<ProductCategoryNavigation>
{
    public ProductCategoryNavigationMap()
    {
        ReadOnly();

        // Set "CategoryId" property as the ID column. Without this, 
        // OpenSession() threw an exception that the configuration was invalid
        Id(x => x.CategoryId);
        Map(x => x.CategoryNodeId);
        Map(x => x.ParentCategoryNodeId);
        Map(x => x.Name);
        Map(x => x.Title);
        Map(x => x.SeoUrl);
        // The column name returned from the sproc is "VisibleInd", 
        // so this is here to map it to the "IsActive" property
        Map(x => x.IsActive).Column("VisibleInd"); 
        Map(x => x.DisplayOrder);
        Map(x => x.ProductCount);
    }
}

对存储过程的调用如下所示:

public List<NavigationViewModel> GetNavigationViewModel(int portalId, int localeId)
{
    const string sql = "EXEC [dbo].[Stored_Procedure_Name] @PortalId=:PortalId, @LocaleId=:LocaleId";
    return _session.CreateSQLQuery(sql)
                .AddEntity(typeof(ProductCategoryNavigation))
                .SetInt32("PortalId", portalId)
                .SetInt32("LocaleId", localeId)
                .List<ProductCategoryNavigation>()
                .Select(x => new NavigationViewModel
                                 {
                                     CategoryId = x.CategoryId,
                                     CategoryNodeId = x.CategoryNodeId,
                                     ParentCategoryNodeId = x.ParentCategoryNodeId,
                                     Name = x.Name,
                                     Title = x.Title,
                                     SeoUrl = x.SeoUrl,
                                     IsActive = x.IsActive,
                                     DisplayOrder = x.DisplayOrder,
                                     ProductCount = x.ProductCount
                                 })
                .ToList();
}

AddEntity 调用说明要将结果映射到哪个 Entity 类,它将使用上面定义的 ProductCategoryNavigationMap:

.AddEntity(typeof(ProductCategoryNavigation))

如果您仔细查看“sql”变量的值,您会看到两个参数:

  1. :PortalId
  2. :LocaleId

这些是通过调用:

.SetInt32("PortalId", portalId)
.SetInt32("LocaleId", localeId)

然后调用.List<ProductCategoryNavigation>()为我们提供了一个 IList,它允许我们使用 LINQ 来投影我们想要的任何东西。在这种情况下,我得到一个 NavigationViewModel 列表,它目前与 ProductCategoryNavigation 相同,但可以根据需要独立于实体进行更改。

我希望这可以帮助其他刚接触 NHibernate 的开发人员!

于 2011-12-24T02:47:57.973 回答
0

假设您已经正确安装了 NHibernate,您将创建一个新类,用于存储您的类映射。

创建一个类,如:

public class PcnMap : ClassMap<ProductCategoryNavigation>
{
   Table("TableName");
   Id( x => x.CategoryId );
   Map( model => model.CategoryNodeId );
   // more like this for all your properties
}

完成设置后,您可以根据需要使用存储库。

请记住,这只是一个基本设置。你的数据库结构越复杂,你的类映射就越复杂。

于 2011-12-23T18:49:52.620 回答