c# - 层次结构中多类的 Fluent NHibernate 每个继承表 (TPH) 映射

标签 c# nhibernate orm fluent-nhibernate mapping

大家好,

我正在努力解决一个 Fluent NHibernate 问题。 我的解决方案中有以下类结构模板:

class OneClass
{
    public virtual string OneProp {get; set;}
}

class TwoClass : OneClass
{
     public virtual string TwoProp {get; set;}
}

class ThreeClass : TwoClass
{
     public virtual string ThreeProp {get; set;}
}

我想为我的类使用一个表继承的层次结构策略,以便在数据库的一个表中包含所有数据。

如何通过 Fluent NHibernate 实现?

我试过以下情况:

<强>1。我为父类添加了鉴别器

 public class OneClassMappingOverride : IAutoMappingOverride<OneClass>
 {
        public void Override(AutoMapping<OneClass> mapping)
        {
            mapping.DiscriminateSubClassesOnColumn("Type");
            mapping.SubClass<OneClass>("OneClass");
            mapping.SubClass<TwoClass>("TwoClass");
            mapping.SubClass<ThreeClass>("ThreeClass");
        }
 }

但我遇到了异常:(XmlDocument)(56,8): XML 验证错误:命名空间“urn:nhibernate-mapping-2.2”中的元素“子类”在命名空间中具有无效的子元素“joined-subclass” 'urn:nhibernate-mapping-2.2'。预期的可能元素列表:'meta, tuplizer, synchronize, property, many-to-one, one-to-one, component, dynamic-component, properties, any, map, set, list, bag, idbag, array, primitive -array, join, subclass, loader, sql-insert, sql-update, sql-delete, resultset, query, sql-query' in namespace 'urn:nhibernate-mapping-2.2'.

因为我有以下为 NHibernate 自动生成的 xml 映射:

<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2">
  <class xmlns="urn:nhibernate-mapping-2.2" name="MyClass.Domain.OneClass, MyClass, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null" table="`OneClass`">
    <cache usage="read-write" />
    <id name="PersistenceId" type="System.Guid, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
      <column name="Id" />
      <generator class="Systematic.Persistence.NHibernate.NHibernateIdGenerator, Systematic.Core, Version=2.0.0.0, Culture=neutral, PublicKeyToken=null" />
    </id>
    <discriminator type="String">
      <column name="Type" />
    </discriminator>
    <version generated="never" name="PersistedVersion" type="System.Int64, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" unsaved-value="0">
      <column name="PersistedVersion" />
    </version>
    <property name="OneProp" type="System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
      <column name="OneProp" length="255" />
    </property>
    <property name="DisplayName" type="System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
      <column name="DisplayName" length="256" index="idx__DisplayName" />
    </property>
    <property name="SystemName" type="System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
      <column name="SystemName" length="256" index="idx__SystemName" not-null="true" />
    </property>
    <property name="Version" type="System.Int64, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
      <column name="Version" not-null="true" />
    </property>
    <property name="Description" type="System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
      <column name="Description" length="10000000" />
    </property>
    <many-to-one class="Systematic.Persistence.PersistenceInfo, Systematic.Api, Version=2.0.0.0, Culture=neutral, PublicKeyToken=null" fetch="join" lazy="false" name="Persistence">
      <column name="Persistence_id" index="idx__Persistence" not-null="true" />
    </many-to-one>
    <subclass name="MyClass.Domain.TwoClass, MyClass, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null">
      <property name="TwoProp" type="System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
        <column name="TwoProp" length="255" />
      </property>
      <joined-subclass name="MyClass.Domain.ThreeClass, MyClass, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null">
        <key>
          <column name="TwoClass_id" />
        </key>
        <property name="ThreeProp" type="System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
          <column name="ThreeProp" length="255" />
        </property>
      </joined-subclass>
    </subclass>
  </class>
</hibernate-mapping>

<强>2。我尝试添加不同的鉴别器

public class OneClassMappingOverride : IAutoMappingOverride<OneClass>
 {
        public void Override(AutoMapping<OneClass> mapping)
        {
            mapping.DiscriminateSubClassesOnColumn("OneType");
        }
 }

 public class TwoClassMappingOverride : IAutoMappingOverride<TwoClass>
 {
        public void Override(AutoMapping<TwoClass> mapping)
        {
            mapping.DiscriminateSubClassesOnColumn("TwoType");
        }
 }

但系统没有任何结果(相同的异常和映射文件)

<强>3。我尝试使用 ClassMap<> 和 SubclassMap<>

public class OneClassMap : ClassMap<OneClass>
{
    public OneClassMap()
    {
        DiscriminateSubClassesOnColumn("Type");

        Id(x => x.Id);

        Map(x => x.OneProp);
    }
}

public class TwoClassMap : SubclassMap<TwoClass>
{
    public TwoClassMap()
    {
        DiscriminatorValue("TwoType");

        Map(x => x.TwoProp);
    }
}

public class ThreeClassMap : SubclassMap<ThreeClass>
{
    public ThreeClassMap()
    {
        DiscriminatorValue("ThreeType");

        Map(x => x.ThreProp);
    }
}

在这种情况下,我有三个表(我的数据库中有 OneClass、TwoClass 和 ThreeClass),我还有以下 xml 映射文件:

<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2">
  <class xmlns="urn:nhibernate-mapping-2.2" name="MyClass.Domain.OneClass, MyClass, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null" table="`OneClass`">
    <cache usage="read-write" />
    <id name="PersistenceId" type="System.Guid, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
      <column name="Id" />
      <generator class="Systematic.Persistence.NHibernate.NHibernateIdGenerator, Systematic.Core, Version=2.0.0.0, Culture=neutral, PublicKeyToken=null" />
    </id>
    <version generated="never" name="PersistedVersion" type="System.Int64, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" unsaved-value="0">
      <column name="PersistedVersion" />
    </version>
    <property name="OneProp" type="System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
      <column name="OneProp" length="255" />
    </property>
    <property name="DisplayName" type="System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
      <column name="DisplayName" length="256" index="idx__DisplayName" />
    </property>
    <property name="SystemName" type="System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
      <column name="SystemName" length="256" index="idx__SystemName" not-null="true" />
    </property>
    <property name="Version" type="System.Int64, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
      <column name="Version" not-null="true" />
    </property>
    <property name="Description" type="System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
      <column name="Description" length="10000000" />
    </property>
    <many-to-one class="Systematic.Persistence.PersistenceInfo, Systematic.Api, Version=2.0.0.0, Culture=neutral, PublicKeyToken=null" fetch="join" lazy="false" name="Persistence">
      <column name="Persistence_id" index="idx__Persistence" not-null="true" />
    </many-to-one>
    <joined-subclass name="MyClass.Domain.TwoClass, MyClass, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null">
      <key>
        <column name="OneClass_id" />
      </key>
      <property name="TwoProp" type="System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
        <column name="TwoProp" length="255" />
      </property>
      <joined-subclass name="MyClass.Domain.ThreeClass, MyClass, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null">
        <key>
          <column name="TwoClass_id" />
        </key>
        <property name="ThreeProp" type="System.String, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089">
          <column name="ThreeProp" length="255" />
        </property>
      </joined-subclass>
    </joined-subclass>
  </class>
</hibernate-mapping>

我不知道如何解决我的问题。也许有人帮我解决这个问题。

非常感谢。

真诚的, 阿列克谢

最佳答案

我试图解决类似的问题。同样在我的例子中,我在一个程序集中有一个类,并且想从这个非抽象基类继承另一个程序集中的一个类。 Fluent NHibernate 不会映射第二个(继承的)类。正如上面的帖子所描述的,似乎只有在

  1. 只有抽象类在不同的程序集中
  2. 继承层次结构中的所有非抽象类都在同一个程序集中

在我看来,这是领域建模的一个非常严重的设计限制。接近 Fluent NHibernate 项目的人可以确认无法从不同程序集中的非抽象类继承吗?或者有人有解决这个问题的方法吗?

谢谢,罗兰。

关于c# - 层次结构中多类的 Fluent NHibernate 每个继承表 (TPH) 映射,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16563530/

相关文章:

c# - 服务器违反了协议(protocol)。 C# 中的 Section=ResponseStatusLine

c# - 在数据流网络中使用 BufferBlock<T> 的好处

使用 ASP.NET MVC 3 的 NHibernate session + 事务

java - Postgres 和 JPA 命名查询 - WHERE 中的任何 (*) 值

c# - 向后按钮跳过收藏夹中的第一个项目

c# - 保持应用程序始终运行

c# - 我如何在 NHibernate 中进行通用字段和组件级更改跟踪?

NHibernate - 为属性/列设置长度属性有什么意义吗?

nhibernate - 如何强制NHibernate不更新集合中的所有对象

inheritance - ORM 继承