我有一个实体Foo,它引用了一个实体Bar:
@Entity
public class Foo {
@OneToOne(cascade = {PERSIST, MERGE, REFRESH}, fetch = EAGER)
public Bar getBar() {
return bar;
}
}
当我持久化一个新的Foo时,它可以引用一个新的Bar或一个现有的Bar。当它引用一个现有的Bar并且该Bar已经被分离(detached),我的JPA提供者(Hibernate)会抛出以下异常:
Caused by: org.hibernate.PersistentObjectException: detached entity passed to persist: com.example.Bar
at org.hibernate.event.def.DefaultPersistEventListener.onPersist(DefaultPersistEventListener.java:102)
at org.hibernate.impl.SessionImpl.firePersist(SessionImpl.java:636)
at org.hibernate.impl.SessionImpl.persist(SessionImpl.java:628)
at org.hibernate.engine.EJB3CascadingAction$1.cascade(EJB3CascadingAction.java:28)
at org.hibernate.engine.Cascade.cascadeToOne(Cascade.java:291)
at org.hibernate.engine.Cascade.cascadeAssociation(Cascade.java:239)
at org.hibernate.engine.Cascade.cascadeProperty(Cascade.java:192)
at org.hibernate.engine.Cascade.cascade(Cascade.java:153)
at org.hibernate.event.def.AbstractSaveEventListener.cascadeBeforeSave(AbstractSaveEventListener.java:454)
at org.hibernate.event.def.AbstractSaveEventListener.performSaveOrReplicate(AbstractSaveEventListener.java:288)
at org.hibernate.event.def.AbstractSaveEventListener.performSave(AbstractSaveEventListener.java:204)
at org.hibernate.event.def.AbstractSaveEventListener.saveWithGeneratedId(AbstractSaveEventListener.java:130)
at org.hibernate.ejb.event.EJB3PersistEventListener.saveWithGeneratedId(EJB3PersistEventListener.java:49)
at org.hibernate.event.def.DefaultPersistEventListener.entityIsTransient(DefaultPersistEventListener.java:154)
at org.hibernate.event.def.DefaultPersistEventListener.onPersist(DefaultPersistEventListener.java:110)
at org.hibernate.event.def.DefaultPersistEventListener.onPersist(DefaultPersistEventListener.java:61)
at org.hibernate.impl.SessionImpl.firePersist(SessionImpl.java:645)
at org.hibernate.impl.SessionImpl.persist(SessionImpl.java:619)
at org.hibernate.impl.SessionImpl.persist(SessionImpl.java:623)
at org.hibernate.ejb.AbstractEntityManagerImpl.persist(AbstractEntityManagerImpl.java:220)
... 112 more
当我确保对Bar的引用是被管理的(附加的)或者在关系中省略CASCADE PERSIST时,一切都很顺利。
但是这两种解决方案都不是百分之百的满意。如果我删除级联持久性,我显然不能再将一个新的Bar的引用与Foo一起持久化。使Bar的引用得到管理需要在持久化之前编写如下代码:
if (foo.getBar().getID() != null && !entityManager.contains(foo.getBar())) {
foo.setBar(entityManager.merge(foo.getUBar()));
}
entityManager.persist(foo);
对于单个Bar来说,这似乎并不是什么大问题,但是如果我需要考虑所有属性,我最终将得到相当可怕的代码,这似乎违背了首先使用ORM的原因。 我可能要像以前一样使用JDBC手动持久化对象图。
当给定现有的Bar引用时,JPA所要做的唯一事情就是取其ID并将其插入保存Foo的表的列中。 当Bar被附加时,它确实执行此操作,但在Bar分离时抛出异常。
我的问题是;为什么它需要Bar被附加? 当Bar实例从分离状态转换为附加状态时,它的ID肯定不会改变,而且这似乎是唯一需要的东西。
这可能是Hibernate中的一个错误吗?还是我漏看了什么?