gpt4 book ai didi

java - JPA/Hibernate - 嵌入属性

转载 作者:塔克拉玛干 更新时间:2023-11-03 03:32:15 24 4
gpt4 key购买 nike

我在映射类的嵌入式属性时遇到问题。我创建了一些类,这些类与我试图说明的类相似。基本上,我有一个使用继承的@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;
}

}

零件编号

@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;

}

}

福特零件编号

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;

}

}

雪佛兰零件编号

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;

}
}

当然这是行不通的,因为 Hibernate 忽略了 Inheritance Hierarchy 并且不喜欢 PartNumber 是抽象的这一事实。 有什么方法可以使用 JPA 或 Hibernate 注释来做到这一点吗?我尝试过使用 @Inheritance JPA 注释。

我无法重构层次结构的“PartNumber”部分,因为原始开发人员希望能够使用 N 个 XXXXPartNumber 类扩展 PartNumber。

有谁知道这样的东西是否会成为 JPA 2.0 或新版本的 Hibernate 的一部分?

最佳答案

不支持组件(例如@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 {
...
}

B) 根据您的示例,所有 PartNumber 后代似乎仅在行为上有所不同(它们不引入任何要存储的新属性)。如果情况确实如此,您可以将 PartNumber 属性加上您自己的鉴别器值(以便您知道要实例化哪个类)映射为 @Embedded 私有(private)属性,并在 Part 类编码/解码适当的子类中具有 get/setPartNumber() 访问器。您甚至可以编写自己的 Hibernate 自定义类型来为您执行此操作(非常简单)。

C) 如果 PartNumber 后代确实在必须存储的属性上有所不同,并且无论出于何种原因将它们映射为实体都是 Not Acceptable ,您可以使用编码/解码它们以将它们编码为字符串(作为 XML 或任何其他符合要求的形式)并存储那。我使用 XStream 正是为了这个目的,我写了一个简单的 Hibernate 类型来配合它。您的零件映射看起来像

@Type(type="xmlBean")
public PartNumber getPartNumber() {
return partNumber;
}
public void setPartNumber(PartNumber partNumber) {
this.partNumber = partNumber;
}

和 PartNumber 后代根本不需要映射。当然,缺点是在数据库中处理 XML 比较麻烦,因此对于您可能需要报告的内容,这可能不是理想的方法。 OTOH,我正在使用它来存储插件设置,它为我节省了很多映射/数据库维护方面的麻烦。

关于java - JPA/Hibernate - 嵌入属性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1119670/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com