У меня есть сущность Foo, которая ссылается на сущность Bar:
@Entity
public class Foo {
@OneToOne(cascade = {PERSIST, MERGE, REFRESH}, fetch = EAGER)
public Bar getBar() {
return bar;
}
}
Когда я сохраняю новый Foo, он может получить ссылку либо на новый Бар, либо на существующий Бар. Когда он получает существующую панель, которая оказывается отсоединенной, мой поставщик 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);
Для отдельного бара это может показаться не таким уж большим делом, но если мне придется принимать во внимание все свойства, подобные этому, я получу довольно ужасный код, который, кажется, в первую очередь побеждает причину использования ORM , С тем же успехом я могу сохранить свой объектный граф вручную, снова используя JDBC.
При наличии существующей ссылки на бар единственное, что нужно сделать JPA, это взять его идентификатор и вставить его в столбец таблицы, содержащей Foo. Он делает именно это, когда Bar присоединен, но выдает исключение, когда Bar отсоединяется.
Мой вопрос; зачем ему нужно прикрепить Бар? Конечно, его идентификатор не изменится, когда экземпляр Bar перейдет из отключенного в присоединенное состояние, и этот идентификатор, похоже, является единственной вещью, необходимой здесь.
Возможно, это ошибка в Hibernate или я что-то упустил?