2009-05-31 36 views
6

我有以下型号:删除抛出“已删除对象将被重新保存通过级联”

<class name="Person" table="Person" optimistic-lock="version"> 
    <id name="Id" type="Int32" unsaved-value="0"> 
    <generator class="native" /> 
    </id> 
    <!-- plus some properties here --> 
</class> 

<class name="Event" table="Event" optimistic-lock="version"> 
    <id name="Id" type="Int32" unsaved-value="0"> 
    <generator class="native" /> 
    </id> 
    <!-- plus some properties here --> 
</class> 

<class name="PersonEventRegistration" table="PersonEventRegistration" optimistic-lock="version"> 
    <id name="Id" type="Int32" unsaved-value="0"> 
    <generator class="native" /> 
    </id> 
    <property name="IsComplete" type="Boolean" not-null="true" /> 
    <property name="RegistrationDate" type="DateTime" not-null="true" /> 
    <many-to-one name="Person" class="Person" column="PersonId" foreign-key="FK_PersonEvent_PersonId" cascade="all-delete-orphan" /> 
    <many-to-one name="Event" class="Event" column="EventId" foreign-key="FK_PersonEvent_EventId" cascade="all-delete-orphan" /> 
</class> 

没有属性指向PersonEventRegistration亲身也不事件。

当我尝试删除PersonEventRegistration的条目,我得到以下错误:

"deleted object would be re-saved by cascade" 

的问题是,我不存储任何其他集合这个对象 - 删除代码如下所示:

public bool UnregisterFromEvent(Person person, Event entry) 
{ 
    var registrationEntry = this.session 
     .CreateCriteria<PersonEventRegistration>() 
     .Add(Restrictions.Eq("Person", person)) 
     .Add(Restrictions.Eq("Event", entry)) 
     .Add(Restrictions.Eq("IsComplete", false)) 
     .UniqueResult<PersonEventRegistration>(); 

    bool result = false; 
    if (null != registrationEntry) 
    { 
     using (ITransaction tx = this.session.BeginTransaction()) 
     { 
      this.session.Delete(registrationEntry); 
      tx.Commit(); 
      result = true; 
     } 
    } 
    return result; 
} 

我在做什么错在这里?

回答

3

据我所知,cascade="all-delete-orphan"属于集合映射元素,而不是many-to-one。你没有显示你的映射的另外两个部分,所以我不能肯定地说,但这可能(可能)是这个问题。

我觉得Person应该是这个样子:

<!-- other properties --> 
<set name="Events" inverse="true" cascade="all-delete-orphan"> 
    <key column="Person_id" /> 
    <one-to-many class="PersonEventRegistration" /> 
</set> 

Event

<!-- similar mapping for Event --> 

PersonEventRegistration

<!-- other properties --> 
<many-to-one name="Person" class="Person" column="PersonId" foreign-key="FK_PersonEvent_PersonId" cascade="delete" <!-- or many ="all" ? --> /> 

实际上,上述可能是相互冲突的级联(这可能是什么你有)。所以真的,我的回答是两件事:

  1. cascade="all-delete-orphan"many-to-one没有意义。
  2. 确保您真正了解如何与您的实体合作,以及它们应如何级联其操作。
+0

我没有注意到这个职位不包含所有格式... 无论如何 - 事件和人物映射不包含有关PersonEventRegistration的任何信息。这个类用于支持多对多映射+一些额外的属性(我没有找到任何例子,其中nhibernate会支持这种情况)。 反正我会尝试cascade =“delete”。 – Greg 2009-06-02 05:33:48

1

尝试取消引用Person和事件中删除:

public bool UnregisterFromEvent(Person person, Event entry) 
    { 
     var registrationEntry = this.session 
      .CreateCriteria<PersonEventRegistration>() 
      .Add(Restrictions.Eq("Person", person)) 
      .Add(Restrictions.Eq("Event", entry)) 
      .Add(Restrictions.Eq("IsComplete", false)) 
      .UniqueResult<PersonEventRegistration>(); 

     bool result = false; 
     if (null != registrationEntry) 
     { 
      using (ITransaction tx = this.session.BeginTransaction()) 
      { 
       registrationEntry.Person = null; 
       registrationEntry.Event = null; 
       this.session.Delete(registrationEntry); 
       tx.Commit(); 
       result = true; 
      } 
     } 
     return result; 
    } 

另外,我不知道,你可以在对象上增加一个限制,我想这个使用别名和ID采写。

  .Add(Restrictions.Eq("Person", person)) 
      .Add(Restrictions.Eq("Event", entry)) 
相关问题