Создание и получение связанных сопоставлений в NHibernate - PullRequest
0 голосов
/ 12 января 2011

Скажем, у меня есть два постоянных класса Cat и Hat.Давайте представим, что отношения между ними двунаправлены, один-ко-многим, так как у Кошки может быть много разных Шляп, и мы можем пройти отображение в обоих направлениях.Чтобы создать связь между экземплярами каждого класса, я должен быть в состоянии и исправить меня, если я ошибаюсь, сделать следующее:

Cat whiskers = new Cat();
Hat redHat = new Hat();
redHat.Id = 1; // normally assigned by generator but this makes example more explicit
whiskers.Hats.Add(redHat);
catService.Save(whiskers);
hatService.Save(redHat);

Учитывая это, я не должен бытьсмог найти Cat, который связан с redHat, выполнив следующее?

Hat hat = hatService.FindById(1); // redHat has Id 1
Cat cat = hat.Cat; // cat should be whiskers

К сожалению, когда я пытаюсь использовать эту технику, в последней строке cat появляется как нулевой объект.Я предполагаю, что что-то упустил в своем отображении один-ко-многим в Cat.hbm.xml, или, по крайней мере, я надеюсь, что это что-то такое простое.Вот как это выглядит в настоящее время:

<set name="Hats" inverse="true">
  <key column="Id"/>
  <one-to-many class="Hat"/>
</set>

У меня нет сопоставления в Hat.hbm.xml, так как добавление одного привело к ошибке "duplicate mapping".Я прочитал документацию Hibernate по сопоставлениям ассоциаций, но все еще не могу заставить это работать.Что я делаю не так?

Ниже приведены полные файлы сопоставления:

Cat.hbm.xml

<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2"
               assembly="Domain"
               namespace="Domain">

  <class name="Cat" table="cats" lazy="true">

    <id name="Id">
      <generator class="increment" />
    </id>

    <property name="Name" />

    <set name="Hats" inverse="true">
      <key column="Id"/>
      <one-to-many class="Hat"/>
    </set>

  </class>

</hibernate-mapping>

Hat.hbm.xml

<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2"
               assembly="Domain"
               namespace="Domain">

  <class name="Hat" table="hats" lazy="true">

    <id name="Id">
      <generator class="increment" />
    </id>

  </class>

</hibernate-mapping>

Edit

Я считаю, что проблема заключалась в том, что я не инициализировал значение ISet<Hat> в классе Cat.Однако я столкнулся с новой проблемой.Когда я пытаюсь получить доступ к ассоциациям, я получаю эту ошибку:

Could not initialize proxy - no Session.

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

public Cat GetRelatedCat(Hat hat)
{
    Cat cat;
    using (ISession session = HibernateUtil.GetSessionFactory().OpenSession())
    {
        cat = hat.Cat;
    }
    return cat;
}

Это приводит к той же ошибке "без сеанса".Я в тупике.

Ответы [ 2 ]

0 голосов
/ 16 января 2011

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

0 голосов
/ 13 января 2011
  1. Вам нужно добавить запись «многие к одному» в вашем hat.hbm. Я думаю, что должно работать.

  2. Тогда вам нужно установить обе стороны ассоциации

    whiskers.Hats.Add (RedHat);

    redHat.Cat = усов;

Я думаю, это должно заставить ваш пример работать.

Причина вашей проблемы "без сеанса" в том, что прокси-сервер коллекции работает только с сеансом, с которым он был создан. Так что в вашем примере это работает только с сеансом, который создал ваш экземпляр Hat. Можно подключить прокси-сервер к другому сеансу, но обычно это не рекомендуется, поскольку он имеет сложные последствия для границ транзакций и согласованности данных.

...