3

我正在设计的模型的一部分是层次结构地理位置。由于有多个层并共享一些信息,我决定使用这样的类层次结构:

public class GeographicNode
{
    public virtual int Id { get; private set; }
    public virtual string Name { get; set; }
    public virtual GeographicNode ParentNode { get; set; }
    public virtual IList<GeographicNode> ChildNodes { get; set; }
}


public class Region : GeographicNode
{
    public virtual int SomeRegionData { get; set; }
}

public class Country : GeographicNode
{
    public virtual int SomeCountryData { get; set; }
}

为了映射这一点,我使用了每类层次结构表的方法。Fluent nHibernate 映射如下所示:

public class GeographicNodeMap : ClassMap<GeographicNode>
{
    public GeographicNodeMap()
    {
        Id(x => x.Id);
        Map(x => x.Name);
        References(x => x.ParentNode);
        HasMany(x => x.ChildNodes).KeyColumn("Id").Cascade.All();

        DiscriminateSubClassesOnColumn("Type");
    }
}

public class RegionMap : SubclassMap<Region>
{
    public RegionMap()
    {
        Map(x => x.SomeRegionData)
    }
}

public class CountryMap : SubclassMap<Region>
{
    public CountryMap()
    {
        Map(x => x.SomeCountryData)
    }
}

这是我的问题:

当我得到一个节点并尝试访问 ParentNode(或子节点)时,它的类型实际上是 GeographicNode,而不是适当的子类。因此,例如,如果我得到 Region 节点并且它的父节点应该是 Country 节点,我无法将 ParentNode 转换为 Country 类。

有没有办法强制 nHibernate 用适当的子类实例化 ParentNode 和 Child 对象?此信息存储在 Type 列中,因此 nHibernate 实例化正确的子类似乎是合理的。

在这里使用继承是否有任何重大问题?使用类层次结构减少了代码量(至少在模型中),但我担心拥有这些隐式而不是显式的关系(例如,一个地区的父级始终是一个国家)会让我以后遇到麻烦.. .

谢谢!

4

2 回答 2

0

由于您将其映射为普通引用 nhibernate 在不加载父级的情况下无法知道类型,因此在延迟加载父级时它不知道其类型并创建基类的代理。两种补救方法:

  • 禁用参考的延迟加载:References(x => x.ParentNode).Not.LazyLoad();
  • 将父类型存储在子记录中:ReferenceAny<(x => x.ParentNode).TypeColumn("parenttype")...然后 NH 将创建正确的代理

我赞成选项 1,因为它很简单,通常对性能没有太大意义

于 2012-08-31T11:47:06.080 回答
0

我在我的博客中写过这个:http: //mikehadlow.blogspot.com/2010/04/nhibernate-get-vs-load-and-sub-types.html

它包括一个特殊的 Cast 函数,可以正确地转换 NHibernate 代理。

但是,最好的解决方案不是强制转换,而是构建正确的多态代码:)

于 2010-10-15T12:19:43.820 回答