Отображение 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 });