10

我有一个简单的 JOINED 文档层次结构:

CREATE TABLE Documents
(
  id INTEGER NOT NULL,
  discriminator ENUM('official','individual','external') NOT NULL,
  file_name VARCHAR(200) NOT NULL,
  PRIMARY KEY (id)
);

CREATE SystemDocuments
(
  id INTEGER NOT NULL,
  binary_data BLOB NOT NULL,
  PRIMARY KEY (id),
  FOREIGN KEY (id) REFERENCES Documents (id)
);

CREATE ExternalDocuments
(
  id INTEGER NOT NULL,
  PRIMARY KEY (id),
  FOREIGN KEY (id) REFERENCES SystemDocuments (id)
);

如您所见,所有子表所做的都是共享 Documents 表中的相同 ID。除此之外,SystemDocuments添加一binary_data列并且ExternalDocuments不添加任何新属性。(另请注意,层次结构中还有另外两个具体的子表,由'official'和表示,'individual'在此不相关。)

以下是上表的映射:

文档.java

@Entity
@Table(name = "Documents")
@Inheritance(strategy = InheritanceType.JOINED)
@DiscriminatorColumn(name = "discriminator", discriminatorType = DiscriminatorType.STRING)
//@DiscriminatorOptions(force = true) // <-- Hibernate 4-specific annotation not inserting discriminator values
public abstract class Document implements Serializable
{
    @Id
    @Column
    protected Integer id;

    @Column(name = "file_name")
    protected String fileName;

    ...
}

系统文档.java

@Entity
@Table(name = "SystemDocuments")
public abstract class SystemDocument extends Document
{
    @Lob
    @Column(name = "binary_data")
    protected byte[] binaryData;

    ...
}

外部文档.java

@Entity
@Table(name = "ExternalDocuments")
@DiscriminatorValue(value = "external")
public class ExternalDocument extends SystemDocument
{
    ...
}

后一个类应该映射到 Documents 的鉴别器列 value 'external'。通过 EntityManager.find 查找实体时,鉴别器被正确返回,实际上是因为我的测试数据的鉴别器已正确插入到数据库中。

现在我使用以下代码通过 JPA 和文件上传器将新文档/文件插入系统:

...

UploadedFile uf = event.getUploadedFile();

// set ID, file name, and binary data
ExternalDocument detachedExternalDocument =
    new ExternalDocument(1234567, uf.getName(), uf.getData());

docService.create(detachedExternalDocument);

但是,在检查数据库时,我可以看到 Hibernate没有'external'鉴别器值插入到Documents表的discriminator列中。

过去有过这方面的问题,请参阅https://hibernate.onjira.com/browse/ANN-140和最近的 Hibernate 4 https://hibernate.onjira.com/browse/HHH-4358,所以机会它应该这样工作吗?

然后我在当前的 Hibernate 4 API Docs 中找到了http://docs.jboss.org/hibernate/core/4.0/javadocs/org/hibernate/annotations/DiscriminatorOptions.html,但它不起作用(参见@DiscriminatorOptions 在文档类)。

如何让 Hibernate 4使用原始注释插入鉴别器?

注意:我不想将鉴别器列映射为常规列。

4

1 回答 1

22

首先,这个问题与 InheritanceType.JOINED 中的 Discriminator重复。

JPA 规范似乎不需要在 JOINED 继承中保留鉴别器值以下是我从 JPA 专家组成员通过电子邮件收到的信息:

该规范不要求使用鉴别器列来实现 JOINED 继承的实现,但是,假设如果指定了@DiscriminatorColumn,那么它将被使用,即值将被写出。我们没有明确声明如果在代码中指定了@DiscriminatorColumn,则必须使用它,就像我们没有明确声明如果指定了@Column 或@JoinColumn,则值必须存储在表中,但是有只有我们可以或应该指定的那么多。在最低层次上,只是假设了某些物理定律和理性。

手头的问题一直是 Hibernate 的问题,请参见此处:

https://hibernate.atlassian.net/browse/ANN-140

拒绝评论:

EJB3 不需要使用带有 JOINED 映射策略的鉴别器。对于需要鉴别器的 JOINED 映射策略的较差实现是允许的。Hibernate 不需要鉴别器,因为 Hibernate 比这些其他劣质实现要好。

最终只有 SINGLE_TABLE 策略需要一个鉴别器列,JOINED可以在没有的情况下实现。Hibernate 当前的问题是,在使用 @DiscriminatorColumn 映射的 JOINED 继承中持久化子实体时会导致数据不一致,即使 JPA 规范建议在将鉴别器与 JOINED 一起使用时保留鉴别器值。在 RFE 中查看更多信息:

https://hibernate.atlassian.net/browse/HHH-6911

于 2011-12-20T13:42:56.000 回答