最近在做将项目中Hibernate 的OR Mapping 转成 JPA 的 EclipseLink 实现。 在Hibernate 中, 如果我们需要多种类型 mapping 到 一张单表的时候, 我们需要配置这个 discriminator 元素。 ?这个又分为两种情形, 如果在这个表中 我们有一个字段来区分不同的类型的时候, 配置?discriminator?元素的时候就比较简单了。我们指定字段名即可, 如下:
?
<discriminator column="TYPE_CD" type="java.lang.String" insert="false" />?
?然后我们需要在各个类型上 使用属性 discriminator-value 来指定它们各自用来区别的值。 ?
?
?另外一种情形是如果我们没有一个固定的字段来区分, ?我们需要在discriminator 元素中配置 formula 公式, 比如:
?
?
<discriminator type="java.lang.String" formula="case when DRUG_LIST_TYPE_ID = 4 then 'A' when DRUG_LIST_TYPE_ID = 8 then 'B' when DRUG_LIST_TYPE_ID = 9 then 'C' when DRUG_LIST_TYPE_ID = 15 then 'MESSAGE' else 'BASE' end" />
?
第一种情形在 JPA 中还是很简单的 在实体基类定义前面加上?
?
?
@Entity@Table(name = "Entry")@Inheritance(strategy=InheritanceType.SINGLE_TABLE)@DiscriminatorColumn(name="TYPE_CD", discriminatorType=DiscriminatorType.STRING,length=20)@DiscriminatorValue("BASE")public class EntryCMP {?
?
然后子类前面加上 ?不同的 ?
? ??@DiscriminatorValue("BLABLA")
?
?
但是像前面的第二种情形,没有固定的字段的时候, 我们在EclipseLink 中采用 annotation 又怎么配置呢。 ? 在它的官方wiki 中发现一个 ClassExtractor 的annoatation , 我们可以这样来定义 基类 :
?
?
@Entity@Table(name = "Entry")@Inheritance(strategy=InheritanceType.SINGLE_TABLE)@ClassExtractor(EntryClassExtractor.class)public class EntryCMP {?
然后再单独定义一个ClassExtractor 类 , 在这个类中来实现对应于Hibernate 的formula 的逻辑:
?
?
package com.bwang.test.domain.cmp;import org.eclipse.persistence.descriptors.ClassExtractor;import org.eclipse.persistence.sessions.Record;import org.eclipse.persistence.sessions.Session;public class EntryClassExtractor extends ClassExtractor { public Class extractClassFromRow (Record row, Session session) { Long typeId = (Long) row.get("TYPE_ID"); if (typeId != null) { if (typeId.equals(4L)) { return EntryTierCMP.class; } if (typeId.equals(9L)) { return EntryMacCMP.class; } if (typeId.equals(8L)) { return EntryDrugSourceCMP.class; } if (typeId.equals(15L)) { return EntryMessageCMP.class; } } return EntryCMP.class; }}
?
可以看到这个ClassExtractor 是EclipseLink 自家的实现, 我在 JPA 2 specification 中是没有找到的。
?
当然对于基类和子类都不要再指定那个?@DiscriminatorValue 了。?
?
比较 Hibernate 和JPA 的实现 , 感觉还是 Hibernate 这块的定义比较强, 在xml 很简单就搞定, 在JPA 中还得专门实现一个类来搞定。 ??
?
?