mai*_*rgs 8 java nhibernate hibernate jpa java-ee
我在映射类的嵌入属性时遇到了麻烦.我创建了一些类似于我试图说明的类.基本上,我有一个使用继承的@Embeddable类层次结构.顶级类"Part Number"只有一个属性,扩展类没有为"Part Number"类添加任何属性,它们只添加了一些验证/逻辑.
这就是我的意思:
部分
@Entity
@Table(name="PART")
public class Part {
private Integer id;
private String name;
private PartNumber partNumber;
@Id
@GeneratedValue(strategy=GenerationType.SEQUENCE)
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
@Column(name="PART_NAME")
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Embedded
public PartNumber getPartNumber() {
return partNumber;
}
public void setPartNumber(PartNumber partNumber) {
this.partNumber = partNumber;
}
}
Run Code Online (Sandbox Code Playgroud)
零件号
@Embeddable
public abstract class PartNumber {
protected String partNumber;
private String generalPartNumber;
private String specificPartNumber;
private PartNumber() {
}
public PartNumber(String partNumber) {
this.partNumber = partNumber;
}
@Column(name = "PART_NUMBER")
public String getPartNumber() {
return partNumber;
}
public void setPartNumber(String partNumber) {
this.partNumber = partNumber;
}
/**
* @param partNumber
* @return
*/
public boolean validate(String partNumber) {
// do some validation
return true;
}
/**
* Returns the first half of the Part Number
*
* @return generalPartNumber
*/
@Transient
public String getGeneralPartNumber() {
return generalPartNumber;
}
/**
* Returns the last half of the Part Number
* which is specific to each Car Brand
*
* @return specificPartNumber
*/
@Transient
public String getSpecificPartNumber() {
return specificPartNumber;
}
}
Run Code Online (Sandbox Code Playgroud)
FORD PARTNUMBER
public class FordPartNumber extends PartNumber {
/**
* Ford Part Number is formatted as 1234-#1234
*
* @param partNumber
*/
public FordPartNumber(String partNumber) {
super(partNumber);
validate(partNumber);
}
/*
* (non-Javadoc)
*
* @see com.test.PartNumber#validate(java.lang.String)
*/
@Override
public boolean validate(String partNumber) {
// do some validation
return true;
}
/*
* (non-Javadoc)
*
* @see com.test.PartNumber#getGeneralPartNumber()
*/
@Override
public String getGeneralPartNumber() {
return partNumber;
}
/*
* (non-Javadoc)
*
* @see com.test.PartNumber#getSpecificPartNumber()
*/
@Override
public String getSpecificPartNumber() {
return partNumber;
}
}
Run Code Online (Sandbox Code Playgroud)
CHEVY PARTNUMBER
public class ChevyPartNumber extends PartNumber {
/**
* Chevy Part Number is formatted as 1234-$1234
*
* @param partNumber
*/
public ChevyPartNumber(String partNumber) {
super(partNumber);
validate(partNumber);
}
/*
* (non-Javadoc)
*
* @see com.test.PartNumber#validate(java.lang.String)
*/
@Override
public boolean validate(String partNumber) {
// do some validation
return true;
}
/*
* (non-Javadoc)
*
* @see com.test.PartNumber#getGeneralPartNumber()
*/
@Override
public String getGeneralPartNumber() {
return partNumber;
}
/*
* (non-Javadoc)
*
* @see com.test.PartNumber#getSpecificPartNumber()
*/
@Override
public String getSpecificPartNumber() {
return partNumber;
}
}
Run Code Online (Sandbox Code Playgroud)
当然这不起作用,因为Hibernate忽略了继承层次结构,并且不喜欢PartNumber是抽象的事实. 有没有办法使用JPA或Hibernate Annotations来做到这一点? 我尝试过使用@Inheritance JPA注释.
我无法重构层次结构中的"PartNumber"部分,因为原始开发人员希望能够使用N多个XXXXPartNumber类扩展PartNumber.
有谁知道这样的事情是否会成为JPA 2.0或新版Hibernate的一部分?
Chs*_*y76 15
不支持组件(例如@Embeddable)继承,很可能永远不会.这有一个很好的理由 - 实体标识符在Hibernate支持的所有继承策略中起关键作用,而组件没有(映射)标识符.
你有三个选择:
A)将PartNumber(及其所有后代)映射为实体.PartNumber可能仍然是抽象的:
@Entity
@Inheritance(strategy=InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name="part_type", discriminatorType=DiscriminatorType.STRING)
public abstract class PartNumber {
...
}
@Entity
@DiscriminatorValue("Ford")
public class FordPartNumber extends PartNumber {
...
}
Run Code Online (Sandbox Code Playgroud)
B)根据您的示例,似乎所有PartNumber后代的行为都不同(它们不会引入任何要存储的新属性).如果确实如此,您可以将PartNumber属性加上您自己的鉴别器值(因此您知道要实例化哪个类)作为@Embedded私有属性,并在Part类marshall/unmarshall适当的子类中使用get/setPartNumber()访问器.您甚至可以编写自己的Hibernate自定义类型来为您执行此操作(这非常简单).
C)如果PartNumber后代在必须存储的属性上有所不同,并且将它们映射为实体是不可接受的,无论出于何种原因,您可以使用marshall/unmarshall将它们作为字符串(作为XML或符合该法案的任何其他内容)并存储它.我正在使用XStream用于这个目的,我写了一个简单的Hibernate类型.您的零件映射看起来像
@Type(type="xmlBean")
public PartNumber getPartNumber() {
return partNumber;
}
public void setPartNumber(PartNumber partNumber) {
this.partNumber = partNumber;
}
Run Code Online (Sandbox Code Playgroud)
并且根本不必映射PartNumber后代.当然,缺点是在数据库中处理XML有点麻烦,因此可能不是您可能需要报告的理想方法.OTOH,我正在使用它来存储插件设置,它为映射/数据库维护节省了很多麻烦.
| 归档时间: |
|
| 查看次数: |
9566 次 |
| 最近记录: |