【问题标题】:Mapping lists of subclasses映射子类列表
【发布时间】:2012-12-13 03:34:30
【问题描述】:

我有一个关于列表的 NHibernate 问题,这些列表被映射为抽象类的子类。

首先是抽象类的映射:

<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2"
                   auto-import="false"
                   assembly="Magma.Core"
                   namespace="Magma.Core.Business">
    <class name="SyndicatePart" table="biz_syndicatepart" abstract="true" lazy="false">
        <id name="Id" column="id">
            <generator class="guid.comb" />
        </id>
        <discriminator column="parttype" not-null="true" />

        <property name="Identifier" column="name" not-null="true" />
        <property name="Share" column="share" not-null="true" />
        <property name="CadasterNumber" column="cadaster_number" not-null="true" />

        <many-to-one name="Account" column="accountid" lazy="proxy" cascade="all" />
        <many-to-one name="Syndicate" column="syndicateid" lazy="proxy" cascade="all" />

        <subclass name="Condo" discriminator-value="condo" lazy="false">
            <property name="OwnerType" column="ownertype" />

                <many-to-one name="Building" column="buildingid" />
            <many-to-one name="Address" column="addressid" />

            <bag name="Tenants" access="field.camelcase-underscore" table="biz_tenant" inverse="true" cascade="all-delete-orphan">
                <key column="syndicatepartid" />
                <one-to-many class="Tenant" />
            </bag>
        </subclass>

        <subclass name="Parking" discriminator-value="park" lazy="false" />
        <subclass name="Locker" discriminator-value="lock" lazy="false" />
    </class>
</hibernate-mapping>

注意子类“Condo”、“Parking”和“Locker”(在我的例子中,只有 Condo 有其他属性)。这是使用这些子类列表的对象映射:

<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2"
                   assembly="Magma.Core"
                   namespace="Magma.Core.Business">
    <class name="Syndicate" table="biz_syndicate" abstract="true" lazy="false">
        <id name="Id" column="id">
            <generator class="guid.comb" />
        </id>
        <discriminator column="orientation" not-null="true" />

        <property name="Name" column="name" not-null="true" />

        <many-to-one name="Manager" column="managerid" cascade="all-delete-orphan" />

        <bag name="Buildings" table="biz_building" inverse="true" cascade="all-delete-orphan">
            <key column="syndicateid" />
            <one-to-many class="Building" />
        </bag>

        <bag name="Parkings" table="biz_syndicatepart" inverse="true" cascade="all-delete-orphan">
            <key column="syndicateid" />
            <one-to-many class="Parking" />
        </bag>

        <bag name="Lockers" table="biz_syndicatepart" inverse="true" cascade="all-delete-orphan">
            <key column="syndicateid" />
            <one-to-many class="Locker" />
            </bag>

        <subclass name="VerticalSyndicate" discriminator-value="vertical" lazy="false" />
        <subclass name="HorizontalSyndicate" discriminator-value="horizontal" lazy="false" />
    </class>
</hibernate-mapping>

每个列表都映射为一个包,指向同一张桌子,但根据列表(公寓、停车场和储物柜)属于不同的类别。

现在解决问题。问题是,当我尝试访问这些列表中的任何一个时,NHibernate 会获取biz_syndicatepart 表中的所有 行,并根据列表将其转换为适当的类。所以假设我在表中有 3 行,如果我访问 Parkings 列表,我将有 3 个停车位。如果我访问Lockers 列表,我将拥有 3 个储物柜!这是为停车列表生成的 SQL:

SELECT parkings0_.syndicateid     as syndicat7_1_,
       parkings0_.id              as id1_,
       parkings0_.id              as id39_0_,
       parkings0_.name            as name39_0_,
       parkings0_.share           as share39_0_,
       parkings0_.cadaster_number as cadaster5_39_0_,
       parkings0_.accountid       as accountid39_0_,
       parkings0_.syndicateid     as syndicat7_39_0_
FROM   biz_syndicatepart parkings0_
WHERE  parkings0_.syndicateid = '2310fcdf-8ab3-48dd-9a75-9f1e00f6f4fd' /* @p0 */

首先,注意双重parkings0_.id。这是正常的吗? parkings0_.syndicateid (选择语句的第一行和最后一行)也是如此。这个我真的不明白。

另外,请注意 no 区分WHERE 子句被插入以指定我想要的列表类型。我假设如果我访问Parkings 列表,我会看到WHERE [discriminator-column] = [discriminator-value],在我的情况下为WHERE parttype = 'park',但它不在语句中,所以这就是返回每一行的原因。

我读到这可能是 NHibernate 中的一个错误(我目前使用的是 3.1 GA 版本),但阅读该错误的描述时,似乎在使用该表时列表的键位于子类表中​​时会发生该错误每个子类策略(加入子类),所以我认为它不适用于我的情况。

有人可以帮我解决这个问题吗?!我的映射文件有问题吗?为什么在 SELECT 中有双 _id,为什么没有鉴别器 WHERE 子句?

【问题讨论】:

    标签: c# .net nhibernate subclass


    【解决方案1】:

    简单的解决方法是在集合映射中添加 where 子句。例如。

        <bag name="Parkings" table="biz_syndicatepart" inverse="true" cascade="all-delete-orphan" where="parttype='park'">
            <key column="syndicateid" />
            <one-to-many class="Parking" />
        </bag>
    

    【讨论】:

    • 事实上,我们通过删除 ' syndicate ' 对象中每个包中的 ' table ' 属性使其工作。我们还将“force”属性添加到“discriminator”属性中。我们不是 100% 知道它为什么不起作用,但现在看来它确实起作用了!我会将您的答案标记为有效答案,因为这是解决问题的简单方法。
    • 答案+1,评论+1(实际上这本身就是一个答案(对我来说是一个首选的答案))
    猜你喜欢
    • 2021-10-30
    • 1970-01-01
    • 1970-01-01
    • 2016-11-07
    • 2018-04-21
    • 2010-10-27
    • 1970-01-01
    • 2018-06-15
    • 2016-10-13
    相关资源
    最近更新 更多