Отображение nhibernate: коллекция с каскадом = "all-delete-orphan" больше не ссылалась
У меня есть некоторые проблемы с моими беглыми сопоставлениями. У меня есть сущность с дочерней коллекцией объектов, например, Event и EventItems.
Если я устанавливаю каскадное сопоставление коллекции в AllDeleteOrphan, я получаю следующую ошибку при сохранении нового объекта в БД:
NHibernate.HibernateException: коллекция с каскадом = "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; }
}
Довольно здесь. Любые советы были высоко оценены.
Чев
Ответы
Ответ 1
Вам нужно сопоставить _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();
}
}
}
Ответ 2
Я не думаю, что принятый ответ - изящный подход. Возможная проблема заключается в том, что Chev считывает события из базы данных и присваивает новый EventItem
список свойству EventItems
. NHibernate выбрасывает это исключение, когда вы просто игнорируете предыдущий список детей и назначаете новый список детей.
Что вам нужно сделать, это
Если вы хотите отбросить старый EventItems
, сделайте это вместо этого:
events.EventItems.Clear();
events.EventItems.Add(new EventItem { blah blah });
Ответ 3
Отметьте это сообщение SO: NHibernate: удалите дочернюю запись из родительской коллекции
Комментарии к принятому ответу имеют схожую проблему.
Вы можете попробовать удалить AsReadOnly
для своего EventItems
, чтобы проверить, является ли это причиной.