gpt4 book ai didi

java - JPA 级联持续存在并且对分离实体的引用会引发 PersistentObjectException。为什么?

转载 作者:IT老高 更新时间:2023-10-28 20:58:50 26 4
gpt4 key购买 nike

我有一个引用实体 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()使其持久化(实际上 - 返回具有相同状态的持久化实例),并合并级联引用,就像您尝试手动执行的那样。

关于java - JPA 级联持续存在并且对分离实体的引用会引发 PersistentObjectException。为什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4294671/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com