Отображение nhibernate: на коллекцию cascade = "all-delete-orphan" больше не ссылались - PullRequest
17 голосов
/ 24 января 2010

У меня есть некоторые пробники с моими беглыми отображениями. У меня есть сущность с дочерней коллекцией сущностей, например, Event и EventItems.

Если я установил для своего каскадного отображения коллекции значение AllDeleteOrphan, я получаю следующую ошибку при сохранении нового объекта в БД: NHibernate.HibernateException: коллекция с cascade = "all-delete-orphan" больше не упоминалась экземпляром объекта-владельца: Core.Event.EventItems

Если я установлю каскад на Все, он будет работать нормально? Ниже приведены мои классы и файлы сопоставления:

 public class EventMap : ClassMap<Event>
{
    public EventMap()
    {
        Id(x => x.Id, "Id")
            .UnsavedValue("00000000-0000-0000-0000-000000000000")
            .GeneratedBy.GuidComb();

        Map(x => x.Name);
        HasMany(x => x.EventItems)
            .Inverse()
            .KeyColumn("EventId")
            .AsBag()
            .Cascade.AllDeleteOrphan();
    }
}

  public class EventItemMap : SubclassMap<EventItem>
{
    public EventItemMap()
    {
         Id(x => x.Id, "Id")
            .UnsavedValue("00000000-0000-0000-0000-000000000000")
            .GeneratedBy.GuidComb();
        References(x => x.Event, "EventId");
    }
}



public class Event : EntityBase
{
    private IList<EventItem> _EventItems;

    protected Event()
    {
        InitMembers();
    }

    public Event(string name)
        : this()
    {
        Name = name;
    }

    private void InitMembers()
    {
        _EventItems = new List<EventItem>();
    }

    public virtual EventItem CreateEventItem(string name)
    {
        EventItem eventItem = new EventItem(this, name);
        _EventItems.Add(eventItem);
        return eventItem;
    }

    public virtual string Name { get; private set; }
    public virtual IList<EventItem> EventItems
    {
        get
        {
            return _EventItems.ToList<EventItem>().AsReadOnly();
        }
        protected set
        {
            _EventItems = value;
        }
    }
}

    public class EventItem : EntityBase
{
    protected EventItem()
    {
    }

    public EventItem(Event @event, string name):base(name)
    {
        Event = @event;
    }

    public virtual Event Event { get; private set; }
}

Довольно тупо здесь. Любые советы с благодарностью.

Чев

Ответы [ 3 ]

24 голосов
/ 24 января 2010

Вам необходимо отобразить _EventItems, используя стратегию доступа, чтобы NHibernate получал доступ к приватному члену, а не к свойству. Вы получаете эту ошибку, потому что ссылка на коллекцию изменяется, когда список копируется в новый список в _EventItems.ToList<EventItem>(). Попробуйте это:

public class EventMap : ClassMap<Event>
{
    public EventMap()
    {
        Id(x => x.Id, "Id")
            .UnsavedValue("00000000-0000-0000-0000-000000000000")
            .GeneratedBy.GuidComb();

        Map(x => x.Name);
        HasMany(x => x.EventItems)
            .Access.PascalCaseField(Prefix.Underscore)
            .Inverse()
            .KeyColumn("EventId")
            .AsBag()
            .Cascade.AllDeleteOrphan();
        }
    }
}
19 голосов
/ 05 мая 2010

Я не думаю, что принятый ответ - элегантный подход. Возможная проблема заключается в том, что Chev читает Events из базы данных, а затем присваивает новый список EventItem свойству EventItems. NHibernate выдает это исключение, когда вы просто игнорируете предыдущий список детей и назначаете новый список детей.

Что вам нужно сделать здесь,

Если вы хотите отказаться от старого EventItems, сделайте это вместо:

events.EventItems.Clear();
events.EventItems.Add(new EventItem { blah blah });
0 голосов
/ 24 января 2010

Проверьте это сообщение SO: NHibernate: Удалить дочернюю запись из родительской коллекции

В комментариях к принятому ответу есть похожая проблема.

Вы можете попробовать удалить AsReadOnly для вашего EventItems, чтобы проверить, является ли это причиной.

...