Отображение списков подклассов - PullRequest
3 голосов
/ 19 июля 2011

У меня проблема с 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 (первая и последняя строка оператора select). Это я действительно не понимаю.

Также обратите внимание, что no различающее предложение WHERE вставлено, чтобы указать, какой тип списка мне нужен. Я бы предположил, что если бы я получил доступ к списку Parkings, я бы увидел WHERE [discriminator-column] = [discriminator-value], в моем случае WHERE parttype = 'park', но его нет в утверждении, поэтому именно поэтому возвращается каждая строка.

Я читал, что это может быть ошибка в NHibernate (в настоящее время я использую версию 3.1 GA), но, читая описание ошибки, кажется, что это происходит, когда ключ списка находится в таблице подклассов при использовании таблицы за стратегию подкласса (присоединенный подкласс), поэтому я не думаю, что она применима к моей ситуации.

Может ли кто-нибудь помочь мне с этим ?! У меня есть проблема с моими файлами сопоставления? Почему двойное _id в SELECT и почему нет условия дискриминатора WHERE?

1 Ответ

2 голосов
/ 20 июля 2011

Простой обходной путь - добавить предложение 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>
...