У меня есть простая СОЕДИНЕННАЯ иерархия документов:
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)
);
Как вы можете видеть, все подтаблицы имеют один и тот же идентификатор из таблицы «Документы». Кроме этого 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;
...
}
SystemDocument.java:
@Entity
@Table(name = "SystemDocuments")
public abstract class SystemDocument extends Document
{
@Lob
@Column(name = "binary_data")
protected byte[] binaryData;
...
}
ExternalDocument.java:
@Entity
@Table(name = "ExternalDocuments")
@DiscriminatorValue(value = "external")
public class ExternalDocument extends SystemDocument
{
...
}
Последний класс должен быть сопоставлен со значением столбца дискриминатора документов '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'
в столбец discriminator
таблицы Documents
.
В прошлом были проблемы с этим, см. https://hibernate.onjira.com/browse/ANN-140 и совсем недавно для Hibernate 4 https://hibernate.onjira.com/browse/HHH-4358, так что, скорее всего, так и должно быть.
Затем я нашел http://docs.jboss.org/hibernate/core/4.0/javadocs/org/hibernate/annotations/DiscriminatorOptions.html в текущих документах API Hibernate 4, но это не работает (см. @DiscriminatorOptions в классе Document).
Как я могу заставить Hibernate 4 вставлять дискриминаторы используя необработанные аннотации?
Примечание. Я не хочу отображать столбец дискриминатора как обычный столбец.