我有一个仅包含2个字段的表。 该表具有由这两个字段形成的复合PK。
使用Netbeans从数据库创建实体bean时,不会像其他具有两个以上字段的表那样自动创建实体bean。
所以我想我需要自己创建实体bean。 创建此实体bean的最佳实践是什么? 是否必须包含COMPOSITE KEY对象?
@Xavi Lopez:他们一起构成了公园。
我不使用NetBeans,因此我不能说任何有关其映射工具的信息。
要映射组合键,有两个选项。您可以
使用PK字段定义一个单独的@Embeddable对象,并将其用作@Entity类中的@EmbeddedId
@Embeddable
public class MyCompositePK {
@Column
private String fieldA;
@Column
private String fieldB;
}
@Entity
public class MyBean {
@EmbeddedId
private MyCompositePK id;
@Column
private String fieldC;
}
用PK字段定义一个未映射的POJO,并将其用作@Entity中的@IdClass。
@Entity
@IdClass(value=ClassAB.ClassABId.class)
public class ClassAB implements Serializable {
private String idA;
private String idB;
@Id
@Column(name="ID_A")
public String getIdA(){ return idA; }
public void setIdA(String idA){ this.idA = idA; }
@Id
@Column(name="ID_B")
public String getIdB(){ return idB; }
public void setIdB(String idB){ this.idB = idB; }
static class ClassABId implements Serializable {
private String idA;
private String idB;
public String getIdA(){ return idA; }
public void setIdA(String idA){ this.idA = idA; }
public String getIdB(){ return idB; }
public void setIdB(String idB){ this.idB = idB; }
// implement equals(), hashcode()
}
}
在此示例中,ClassABId是静态内部类,仅为方便起见。
Pascal Thivent对以下问题的出色回答也解释了这些选项:如何使用Hibernate映射组合键?
这个相关的问题讨论了这些方法之间的区别:我应该使用哪种注释:@IdClass或@EmbeddedId。请注意,使用@IdClass方法重复了字段的声明。
无论如何,我认为创建两个类没有其他选择。这就是为什么我问这个问题:映射仅包含复合PK而没有@IdClass或@EmbeddedId的类。似乎有一个特定于休眠的功能。
附带说明一下,如果您可以控制数据库结构,则还可以考虑避免使用复合键。出于某些原因。
假定对象仅包含两个主键字段,它可以是它自己的IdClass吗?
@TomAnderson那很有趣。 我从来没有试过。 似乎有人成功了,但没有成功。 规范似乎没有对此发表意见。 可能取决于实现。
@TomAnderson:我不这么认为,因为ID类必须仅基于复合键值来实现其equals()方法。
谢谢@XaviLópez。您的解释修正了我的代码,该代码被@Tom Anderson声明为自己的IdClass。当我声明自己的IdClass具有2个@Id列时,获取该实体列表的JPA查询正在返回结果列表中的预期" n"个项目,但该列表中的每个项目均为null。但是,预期大小为" n"。更改为较少重构的静态内部类后,它可以返回正确的结果集。
根据我的说法:引用IdClass的自引用将不起作用,因为自已已经是一个实体,并且需要在持久性上下文中。如果我也有相同类型的主键对象,那么在持久性上下文中将有两个"相同"对象。因此,不应允许它。因此,我们不应使用自引用@IdClass。创建一个静态内部类作为较少干扰的主键类型。