当前位置: 首页 > 面试题库 >

JPA级联仍然存在,对分离实体的引用将引发PersistentObjectException。为什么?

锺离嘉茂
2023-03-14
问题内容

我有一个引用实体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%令人满意的。如果我删除了级联持久化,显然我不能再以新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肯定不会更改,并且该ID似乎是这里唯一需要的东西。

这可能是Hibernate中的错误,还是我缺少了什么?


问题答案:

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

foo = entityManager.merge(foo);

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



 类似资料:
  • 我有两个实体类Organization和User具有一对多关系(为了简单起见,Groovy代码!): 当我为一个新的组织创建一个新的用户时,一切都运行良好: 但当我使用一个现有的组织,将其设置为新用户的组织属性时,我会得到一个错误: 将导致: org.hibernate.PersistentObjectException:传递给Persist:Organization的分离实体 我知道不能对一个分

  • 我在尝试使用Spring Data JPA和Hibernate作为JPA提供程序执行批插入操作时遇到了问题。 我有以下方法在我的服务。这就是抛出异常的地方。 Institute是一个实例变量,并提前获取。 我还将实体设置为级联、持久和合并。 为什么这段代码起作用 而此代码不

  • 我仍然被那些为了支持移动和转发而发明的规则弄糊涂了。有一件事我还不太确定: 转发引用是否只是rvalue引用(应用了引用折叠规则)? 如果它是一个rvalue引用,那么为什么函数: 不仅接受rvalues,还接受lvalues?

  • 我有一个JPA持久化的对象模型,它包含多对一的关系:一个有许多。一个有一个。 下面是代码片段: 我能够创建对象,向其添加事务,并正确地持久化对象。但是,当我创建一个事务,使用一个现有的已经持久化的帐户,并持久化该事务时,我会得到一个异常: 原因:org.hibernate.PersistentObjectException:已分离的实体传递给Persist:com.PaulSanwald.Acco

  • 问题内容: 我有一个JPA持久对象模型,其中包含多对一关系:一个具有许多。A 有一个。 这是一段代码: 我能够创建一个对象,向其中添加事务,并正确地持久保存该对象。但是,当我 使用现有的已经持久化的Account 创建一个事务并持久化 该Transaction时 ,出现一个异常: 引起原因:org.hibernate.PersistentObjectException:传递给持久对象的分离实体:c

  • 我使用的是Spring Boot 4、Hibernate和JPA注释。我遇到了这个错误 组织。冬眠PersistentObjectException:传递给persist的分离实体。 我试着在互联网上搜索,但没有找到正确的答案。我尝试使用而不是,但没有成功。此外,我认为在更新资源时使用。 这是我的代码: ` `积垢服务 ` ` 存储库 ` '抽象存储库 ` ` 工厂` '支持形式 ` ` 控制器