更改保留其ID的实体类型

art*_*emb 9 java persistence hibernate

我使用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
}
Run Code Online (Sandbox Code Playgroud)

我有一个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);
Run Code Online (Sandbox Code Playgroud)

上面的代码失败了

org.hibernate.PersistentObjectException: detached entity passed to persist: C
Run Code Online (Sandbox Code Playgroud)

有可能吗?

Chs*_*y76 12

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工具执行它.

因此,该算法是:

  1. 如果它在那里从会话中逐出 B(这很重要)
  2. 执行您的命名查询.
  3. 使用前B的id加载现在已知的C语言.
  4. 根据需要更新/设置属性.
  5. 坚持C.