JPA 级联持久化,对分离实体的引用会引发 PersistentObjectException。为什么?

2022-09-01 09:04:22

我有一个引用实体Bar的实体Foo:

@Entity
public class Foo {

    @OneToOne(cascade = {PERSIST, MERGE, REFRESH}, fetch = EAGER)
    public Bar getBar() {
        return bar;
    }
}

当我保留一个新的 Foo 时,它可以获得对新 Bar 或现有 Bar 的引用。当它获得一个碰巧被分离的现有 Bar 时,我的 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 的引用是托管的(附加的),或者当我省略关系中的级联 PERSIST 时,一切都很顺利。

然而,这两种解决方案都不是100%令人满意的。如果我删除级联持久性,我显然不能再使用引用新柱的 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 实例从分离状态转换为附加状态时,其 ID 不会更改,并且该 ID 似乎是这里唯一需要的。

这可能是Hibernate中的一个错误,还是我错过了什么?


答案 1

在这种情况下,您可以使用以下命令来代替:merge()persist()

foo = entityManager.merge(foo); 

当应用于新实例时,使其持久化(实际上 - 返回具有相同状态的持久化实例),并合并级联引用,就像您尝试手动执行的那样。merge()


答案 2

如果我理解正确,你只需要引用,以允许新人在持久化时具有外键值(对现有)。在调用上有一个 JPA 方法,对于这种情况,它可能对您有用。该方法与此方法类似,不同之处在于它不会费心返回托管实例(的 ),除非它碰巧已缓存在持久性上下文中。它将返回一个代理对象,该对象将满足您的外键需求,以便持久化该对象。我不确定这是否是你所希望的那种解决方案,但试一试,看看这是否适合你。BarFooBarEntityManagergetReference()getReference()find()BarFoo

我还从您的代码中注意到,您正在通过注释getter方法(对于关系)来使用“属性”样式访问而不是“字段”样式访问。有什么原因吗?出于性能原因,建议您对成员而不是 getter 进行注释。对于 JPA 提供程序来说,直接访问字段应该比通过 getter 和 setter 更有效。Bar

编辑:

正如其他人所提到的,使用级联 merge() 将保留新实体,以及合并修改后的实体并重新附加与 MERGE 级联选项有关系的分离实体。使用“持久化级联”选项不会重新附加任何内容或合并任何内容,并且应该在您想要的行为时使用。