NHibernate не сохранил отношения многие ко многим - PullRequest
7 голосов
/ 30 декабря 2008

В настоящее время я использую NHibernate в качестве слоя доступа к данным, используя Fluent NHibernate для создания файлов сопоставления для меня. У меня есть два класса, TripItem и TripItemAttributeValue, которые имеют отношение многих ко многим.

Отображение выглядит следующим образом:

public class TripItemMap : ClassMap<TripItem2>
{
    public TripItemMap()
    {
        WithTable("TripItemsInt");
        NotLazyLoaded();

        Id(x => x.ID).GeneratedBy.Identity().WithUnsavedValue(0);
        Map(x => x.CreateDate, "CreatedOn").CanNotBeNull();
        Map(x => x.ModifyDate, "LastModified").CanNotBeNull();

        /* snip */

        HasManyToMany<TripItemAttributeValue>(x => x.Attributes).AsBag()
            .WithTableName("TripItems_TripItemAttributeValues_Link")
            .WithParentKeyColumn("TripItemId")
            .WithChildKeyColumn("TripItemAttributeValueId")
            .LazyLoad();
    }
}

public class TripItemAttributeValueMap : ClassMap<TripItemAttributeValue>
{
    public TripItemAttributeValueMap()
    {
        WithTable("TripItemAttributeValues");

        Id(x => x.Id).GeneratedBy.Identity();
        Map(x => x.Name).CanNotBeNull();

        HasManyToMany<TripItem2>(x => x.TripItems).AsBag()
            .WithTableName("TripItems_TripItemAttributeValues_Link")
            .WithParentKeyColumn("TripItemAttributeValueId")
            .WithChildKeyColumn("TripItemId")
            .LazyLoad();
    }
}

В какой-то момент моего приложения я извлекаю существующие атрибуты из базы данных, добавляю их в tripItem.Attributes, затем сохраняю объект tripItem. В конце концов, TripItems_TripItemAttributeValues_Link никогда не получает никаких новых записей, в результате чего отношения не сохраняются.

Если это помогает, это файлы сопоставления, сгенерированные Fluent NHibernate для этих классов:

<?xml version="1.0" encoding="utf-8"?>
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2" default-lazy="true" assembly="ETP.Core" namespace="ETP.Core.Domain">
  <class name="TripItem2" table="TripItemsInt" xmlns="urn:nhibernate-mapping-2.2" lazy="false">
    <id name="ID" column="ID" type="Int32" unsaved-value="0">
      <generator class="identity" />
    </id>
    <property name="CreateDate" column="CreatedOn" type="DateTime" not-null="true">
      <column name="CreatedOn" />
    </property>
    <property name="ModifyDate" column="LastModified" type="DateTime" not-null="true">
      <column name="LastModified" />
    </property>
    <bag name="Attributes" lazy="true" table="TripItems_TripItemAttributeValues_Link">
      <key column="TripItemId" />
      <many-to-many column="TripItemAttributeValueId" class="ETP.Core.Domain.TripItemAttributeValue, ETP.Core, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null" />
    </bag>
  </class>
</hibernate-mapping>

и

<?xml version="1.0" encoding="utf-8"?>
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2" default-lazy="true" assembly="ETP.Core" namespace="ETP.Core.Domain">
  <class name="TripItemAttributeValue" table="TripItemAttributeValues" xmlns="urn:nhibernate-mapping-2.2">
    <id name="Id" column="Id" type="Int32">
      <generator class="identity" />
    </id>
    <property name="Name" column="Name" length="100" type="String" not-null="true">
      <column name="Name" />
    </property>
    <bag name="TripItems" lazy="true" table="TripItems_TripItemAttributeValues_Link">
      <key column="TripItemAttributeValueId" />
      <many-to-many column="TripItemId" class="ETP.Core.Domain.TripItem2, ETP.Core, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null" />
    </bag>
  </class>
</hibernate-mapping>

Что я здесь не так делаю?

Ответы [ 9 ]

8 голосов
/ 21 января 2009

@ efdee

У меня была та же проблема, и я потратил на это почти два дня. У меня были отношения многие ко многим, и таблица ссылок также не обновлялась. Я новичок в NHibernate, просто пытаюсь выучить его, поэтому примите все, что я говорю, с долей соли.

Хорошо, оказалось, что это не Fluent NHibernate, ни отображение, но я не понимаю, как NHibernate работает со многими ко многим. В отношении «многие ко многим», если коллекции на обоих объектах не заполнены, NHibernate не сохраняет данные в таблицу ссылок.

Допустим, у меня есть эти сущности в отношениях многих ко многим:


partial class Contact
{
   public string ContactName {get; set;}
   public IList Locations {get; set;}

}

partial class Location
{
   public string LocationName {get; set;}
   public string LocationAddress {get;set;}
   public IList Contacts {get;set;}
}

при добавлении в местоположение в Contact.Locations, я должен убедиться, что контакт также присутствует внутри location.Contacts.

чтобы добавить местоположение, у меня есть этот метод внутри моего класса Contact.


public void AddLocation(Location location)
        {
            if (!location.Contacts.Contains(this))
            {
                location.Contacts.Add(this);
            }
            Locations.Add(location);
        }

Кажется, это решило мою проблему, но, как я уже сказал, я просто беру NHibernate и изучаю его, возможно, есть лучший способ. Если у кого-то есть лучшее решение, пожалуйста, напишите.

Этот пост указал мне проверить обе коллекции: http://www.coderanch.com/t/217138/Object-Relational-Mapping/link-table-of-ManyToMany-annotation

7 голосов
/ 17 октября 2009

Вызовите Session.Flush () или используйте транзакцию.

2 голосов
/ 05 марта 2011

У меня тоже была такая же проблема - данные объединения для многих ко многим не сохранились. Я скопировал отображение из другого отношения многие-к-одному (изменил его для отношения многие-ко-многим), но сохранил атрибут inverse = "true". Когда я удалил этот атрибут, проблема была решена.

2 голосов
/ 02 января 2009

Я не уверен, как вы делаете это с Fluent NHibernate, но вам нужно установить опцию Каскад на сумке (TripItems). Как обычно, У Айенде есть полезный пост о каскадных опциях .

Из быстрого Google, я бы посоветовал вам попробовать:

HasManyToMany<TripItem2>(x => x.TripItems).AsBag()
        .WithTableName("TripItems_TripItemAttributeValues_Link")
        .WithParentKeyColumn("TripItemAttributeValueId")
        .WithChildKeyColumn("TripItemId")
        .LazyLoad()
/*-->*/ .Cascade.All(); /*<-- this is the bit that should make it work */
1 голос
/ 12 июня 2009

У меня есть, и я надеюсь, что это поможет кому-то еще там. Проблема в том, что у меня было обратное = 'true' на обеих сумках. Если вы прочтете приведенный ниже отрывок, вы заметите, что обратное значение должно быть установлено в true только для одного из пакетов:

Обратите внимание на использование inverse = "true". Еще раз, этот параметр указывает NHibernate игнорировать изменения, внесенные в коллекцию категорий, и использовать другой конец ассоциации - коллекцию элементов - в качестве представления, которое должно быть синхронизировано с базой данных.

1 голос
/ 12 июня 2009

У меня точно такая же проблема, но я использую NHibernate.JetDriver. Я пытался использовать рекомендованный ответ без какого-либо успеха. Кто-нибудь знает, есть ли ограничение NHibernate.JetDriver по отношению ко многим ко многим?

Вот мои файлы hbm на случай, если кто-то заинтересовался их просмотром:

<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2" namespace="Ace.Docs.Core.Domain" assembly="Ace.Docs.Core">
<class name="Ace.Docs.Core.Domain.Address, Ace.Docs.Core" table="Addresses" lazy="true">
    <id name="Id" column="ID">
        <generator class="identity" />
    </id>
    <property name="Address1" column="Address1" />
    <property name="Address2" column="Address2" />
    <property name="City" column="City" />
    <property name="EmailAddress" column="EmailAddress" />
    <property name="Phone1" column="Phone1" />
    <property name="Phone2" column="Phone2" />
    <property name="PostalCode" column="PostalCode" />
    <property name="StateOrProvince" column="StateOrProvince" />
    <many-to-one name="AddressTypeMember" column="AddressTypeID" class="AddressType" />
    <bag name="HasPersonalInfo" table="Link_PersonalInfo_Addresses" lazy="true" cascade="save-update" inverse="true" >
        <key column="AddressID"></key>
        <many-to-many column="PersonalInfoID" class="PersonalInfo" />
    </bag>
</class>

<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2" namespace="Ace.Docs.Core.Domain" assembly="Ace.Docs.Core">
<class name="Ace.Docs.Core.Domain.PersonalInfo, Ace.Docs.Core" table="PersonalInfo" lazy="true">
    <id name="Id" column="ID">
        <generator class="identity" />
    </id>
    <property name="Prefix" column="Prefix" />
    <property name="FirstName" column="FirstName" />
    <property name="MiddleName" column="MiddleName" />
    <property name="LastName" column="LastName" />
    <property name="SIN" column="SIN" />
    <property name="Birthdate" column="Birthdate" />
    <property name="Note" column="Notes" />
    <bag name="HasAddress" table="Link_PersonalInfo_Addresses" lazy="true" cascade="save-update" inverse="true" >
        <key column="PersonalInfoID"></key>
        <many-to-many column="AddressID" class="Address" />
    </bag>
</class>

1 голос
/ 02 января 2009

Дэвид Кемп имеет право: вы хотите добавить каскад в вашу сумку.

Я всегда вручную редактировал (и создавал вручную) файлы сопоставления, поэтому моя естественная склонность - поместить его туда. Вы можете сделать это следующим образом:

<bag name="TripItems" lazy="true" table="TripItems_TripItemAttributeValues_Link" cascade="all">
  <key column="TripItemAttributeValueId" />
  <many-to-many column="TripItemId" class="ETP.Core.Domain.TripItem2, ETP.Core, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null" />
</bag>

Я обнаружил, что поддержание чистоты моих классов и все, что связано с nHibernate в файле .hbm.xml, делает мое решение более чистым. Таким образом, если я хочу использовать новое программное обеспечение ORM, я просто заменяю файлы отображения и не переписываю классы. Мы используем наши модульные тесты для тестирования классов и обеспечения тестируемости для xml, хотя я в некотором роде похож на методы Fluent NHibernate.

0 голосов
/ 20 июня 2009

Я тоже боролся с этим и пришел по совершенно другой причине для моих неприятностей. В моем примере, если бы у меня был объект без каких-либо отношений «многие ко многим», я мог бы просто вызвать saveOrUpdate, и все было бы хорошо. Но если бы у меня были какие-то отношения «многие ко многим», я должен был убедиться, что мой вызов saveOrUpdate был внутри BeginTransaction и CommitTransaction. Я очень плохо знаком с Nhibernate, поэтому прошу прощения, если это очевидно. Но это было не для меня.

0 голосов
/ 03 января 2009

Боюсь, что Cascade.All () на самом деле не является решением моей проблемы - это была одна из тех вещей, которые я попробовал. Проблема не в том, что элементы, добавленные в коллекцию, не сохраняются - они уже находятся в базе данных на момент добавления в коллекцию. Просто записи в таблице ссылок не создаются. Кроме того, я думаю, что Cascade.All () также приведет к удалению дочерних элементов, что нежелательно в моем сценарии. Я пытался использовать Cascade.SaveUpdate (), но, как я уже говорил, это решает что-то, что не является моей проблемой: -)

Однако, чтобы убедиться, я попробую это решение и сообщу вам результат.

Что касается поддержания чистоты классов, то это 100% случай с Fluent NHibernate. Созданные вами сопоставления классов - это файлы кода C #, которые идут вместе с вашими классами сущностей, почти как файлы .hbm.xml.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...