我使用hibernate作为持久层.有两个实体位于同一个表中,扩展了一个具有单表继承策略的超类.
@Entity @Inheritance(strategy = InheritanceType.SINGLE_TABLE) public abstract class A { @Id @GeneratedValue protected Long id; // some common fields for B and C } @Entity public class B extends A { // B-specific fields } @Entity public class C extends A { // C-specific fields }
我有一个ID为4的B实例.如何将此实例的类型更改为C保留其ID(4)?
B b = em.find(B.class, 4L); C c = convertToC(b); c.setId(b.getId()); em.remove(b); em.persist(c);
上面的代码失败了
org.hibernate.PersistentObjectException: detached entity passed to persist: C
有可能吗?
Hibernate试图使持久性尽可能透明 - 这意味着它试图遵循与普通Java对象相同的原则.现在,用Java重写你的问题,你会得到:
如何将B类的实例转换为(不兼容的)C类的实例?
你知道答案 - 你做不到.您可以创建C 的新实例并复制必要的属性,但B将始终为B,而不是C.因此,原始问题的答案是 - 无法通过JPA或Hibernate API完成.
然而,与普通Java不同,使用Hibernate可以作弊:-) InheritanceType.SINGLE_TABLE
使用映射,@DiscriminatorColumn
并且为了将B转换为C,您需要将其值从B指定的任何内容更新为C的任何指定.诀窍是 - 你不能这样做使用Hibernate API; 你需要通过纯SQL来做到这一点.但是,您可以将此更新语句映射为命名SQL查询,并使用Hibernate工具执行它.
因此,该算法是:
如果它在那里从会话中逐出 B(这很重要)
执行您的命名查询.
使用前B的id加载现在已知的C语言.
根据需要更新/设置属性.
坚持C.