Смешивание наследования и древовидной структуры с помощью Fluent nHibernate - PullRequest
3 голосов
/ 28 февраля 2010

Часть модели, которую я разрабатываю, представляет собой иерархию географических местоположений. Поскольку существует несколько слоев и они делятся некоторой информацией, я решил использовать иерархию классов, например:

public class GeographicNode
{
    public virtual int Id { get; private set; }
    public virtual string Name { get; set; }
    public virtual GeographicNode ParentNode { get; set; }
    public virtual IList<GeographicNode> ChildNodes { get; set; }
}


public class Region : GeographicNode
{
    public virtual int SomeRegionData { get; set; }
}

public class Country : GeographicNode
{
    public virtual int SomeCountryData { get; set; }
}

Чтобы отобразить это, я использую метод иерархии таблиц на класс. Отображение Fluent nHibernate выглядит следующим образом:

public class GeographicNodeMap : ClassMap<GeographicNode>
{
    public GeographicNodeMap()
    {
        Id(x => x.Id);
        Map(x => x.Name);
        References(x => x.ParentNode);
        HasMany(x => x.ChildNodes).KeyColumn("Id").Cascade.All();

        DiscriminateSubClassesOnColumn("Type");
    }
}

public class RegionMap : SubclassMap<Region>
{
    public RegionMap()
    {
        Map(x => x.SomeRegionData)
    }
}

public class CountryMap : SubclassMap<Region>
{
    public CountryMap()
    {
        Map(x => x.SomeCountryData)
    }
}

Вот мой вопрос:

Когда я получаю узел и пытаюсь получить доступ к ParentNode (или дочерним элементам), его тип на самом деле - GeographicNode, а не соответствующий подкласс. Так, например, если я получаю узел Region, а его родитель должен быть узлом Country, я не могу привести ParentNode к классу Country.

Есть ли способ заставить nHibernate создавать экземпляры объектов ParentNode и Child с их соответствующим подклассом? Эта информация хранится в столбце Тип, поэтому было бы разумно, чтобы nHibernate создал правильный подкласс.

Также есть ли какие-либо серьезные проблемы с использованием наследования здесь? Использование иерархии классов уменьшает объем кода (по крайней мере, в модели), но я беспокоюсь, что наличие этих неявных, а не явных отношений (например, родитель региона всегда является страной) может привести к проблемам позже. .

Спасибо!

Ответы [ 2 ]

0 голосов
/ 31 августа 2012

, поскольку вы отобразили его как обычную ссылку, nhibernate не может знать тип, который загружает родительский элемент, поэтому при отложенной загрузке родительского элемента он не знает его тип и создает прокси-сервер базового класса. Два варианта, чтобы исправить это:

  • отключить отложенную загрузку ссылки: References(x => x.ParentNode).Not.LazyLoad();
  • сохранить родительский тип в записи о потомке: ReferenceAny<(x => x.ParentNode).TypeColumn("parenttype")... тогда NH создаст правильный прокси

Я бы предпочел вариант 1, поскольку он прост и, как правило, не имеет большого значения для производительности

0 голосов
/ 15 октября 2010

Я написал об этом в своем блоге здесь: http://mikehadlow.blogspot.com/2010/04/nhibernate-get-vs-load-and-sub-types.html

Включает специальную функцию Cast, которая правильно разыгрывает прокси NHibernate.

Тем не менее, лучшее решение - не приведение, а правильное построение полиморфного кода:)

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