首先看看什么是联合主键:
联合主键一般以表中的两个或两个以上的元素作为主键。
一般的联合主键,在数据库中设置就可以了。但是,如果你想用hibernate对主键做映射的话,则需要用面向对象的方法去操作它。
那么,如何用面向对象的方法去得到它呢?而又如何以面向对象的方式分配主键呢?
有两种方式:
XML配置联合主键和Annotation配置联合主键。
XML配置联合主键:
操作方式:
把多个主键封装在一个类中。
如Student类,如果以id和name为主键,那么就写一个类(StudentPK),将id和name作为成员变量(private),设置getter和setter方法。
在Student.hbm.xml中设置联合主键:
<!--设置联合主键--> <composite-id name="pk" class="cn.edu.hpu.model.StudentPK"> <key-property name="id"></key-property> <key-property name="name"></key-property> </composite-id> <property name="name"></property> <property name="age"></property>此方法要继承implements java.io.Serializable接口,实现序列化。
序列化是干什么的?序列化是可以把对象写到硬盘上,或可以通过网络传输对象。
为什么要实现序列化呢?
1、Student在数据库表中可能存在多条记录,这多条记录放在内存中就是多个Student对象,每一个对象都有一个PK联合主键。如果以后系统要做集群,好多服务器要同时对外提供服务,某台服务器出问题了,可以把这台服务器的对象传给另外一个服务器,这时候要实现序列化。
2、加入内存满了,可以使用虚拟内存,就是把硬盘上的一部分空间作为内存来使用。这个时候要把内存放在硬盘中,就要实现序列化。
联合主键类(StudentPK):
package cn.edu.hpu.model; public class StudentPK implements java.io.Serializable{ private int id; private String name; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } //联合主键的实体类一定要重写Object类的equals()和hsahCode()方法 //一系列对象放到内存中,是要做区分的,所以为了保证唯一性,要重写equals()和hsahCode()方法 //下面是改写的方法 @Override public boolean equals(Object obj) { if(obj instanceof StudentPK){ StudentPK o=(StudentPK)obj; if(this.id==o.getId()&&this.name.equals(o.getName())){ return true; } } return false; } //为什么要改写hashCode?hashCode是干嘛的? //如果说我们这个对象被装在内存的哈希表里面,查内容是否相同的时候首先查它的哈希表 //哈希表底层是由数组(往往是链表)实现的。 //如果哈希码相同,对象会被装在同一个位置。 /*假如我们将来写了一个程序,我们要把一系列 * 的Student对象装到哈希表里,这个时候我们要计算Student的哈希码才能 * 装到哈希表里。我不会直接计算Student的哈希码,因为数据库是以主键 * 来区分对象的,所以我应该把主键的哈希码计算出来。 * 假设哈希码为2,我们把对象存到为2的位置。当我们找Student * 的时候,在找哈希码为2的学生,并找里面equals的对象 * */ @Override public int hashCode() { return this.name.hashCode(); } }
Annotation配置联合主键:
下面是定义联合主键的几种方法:
1.将组件类注解为@Embeddable,并将组件的属性注解为@Id。
2.将组件的属性注解为@EmbeddedId。
3.将类注解为@IdClass,并将该实体中所有属于主键的属性都注解为@Id。
方法1:
实体类Teacher:
package cn.edu.hpu.model; import java.util.Date; import javax.persistence.Column; import javax.persistence.Entity; import javax.persistence.EnumType; import javax.persistence.Enumerated; import javax.persistence.GeneratedValue; import javax.persistence.GenerationType; import javax.persistence.Id; import javax.persistence.Table; import javax.persistence.Temporal; import javax.persistence.TemporalType; //‘@+语句’就叫做注解 //@Entity表示这家伙是个实体类 //@Table(name="_teacher") //指定实体类对应的表名(如果没有这个表hibernate会自动创建) @Entity @Table(name="_teacher") public class Teacher { private teacherPK PK; private String title; private Date brithday; private ZhiCheng zhicheng; @Id public teacherPK getPK() { return PK; } public void setPK(teacherPK pK) { PK = pK; } @Enumerated(EnumType.STRING) public ZhiCheng getZhicheng() { return zhicheng; } public void setZhicheng(ZhiCheng zhicheng) { this.zhicheng = zhicheng; } @Temporal(TemporalType.DATE) public Date getBrithday() { return brithday; } public void setBrithday(Date brithday) { this.brithday = brithday; } @Column(name="_title") public String getTitle() { return title; } public void setTitle(String title) { this.title = title; } }
//联合主键类teacherPK:
package cn.edu.hpu.model; import javax.persistence.Embeddable; //@Embeddable表示是另外一个类的一部分 @Embeddable public class teacherPK implements java.io.Serializable{ private int id; private String name; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } @Override public boolean equals(Object obj) { if(obj instanceof StudentPK){ StudentPK o=(StudentPK)obj; if(this.id==o.getId()&&this.name.equals(o.getName())){ return true; } } return false; } @Override public int hashCode() { return this.name.hashCode(); } }
//运行测试类后,数据库自己去建立表,而且你会发现id和name都是主键
方法2:
只需要在实体类的主键get方法上加:@EmbeddedId
package cn.edu.hpu.model; import java.util.Date; import javax.persistence.Column; import javax.persistence.EmbeddedId; import javax.persistence.Entity; import javax.persistence.EnumType; import javax.persistence.Enumerated; import javax.persistence.GeneratedValue; import javax.persistence.GenerationType; import javax.persistence.Id; import javax.persistence.Table; import javax.persistence.Temporal; import javax.persistence.TemporalType; @Entity @Table(name="_teacher") public class Teacher { private teacherPK PK; private String title; private Date brithday; private ZhiCheng zhicheng; @EmbeddedId public teacherPK getPK() { return PK; } public void setPK(teacherPK pK) { PK = pK; } @Enumerated(EnumType.STRING) public ZhiCheng getZhicheng() { return zhicheng; } public void setZhicheng(ZhiCheng zhicheng) { this.zhicheng = zhicheng; } @Temporal(TemporalType.DATE) public Date getBrithday() { return brithday; } public void setBrithday(Date brithday) { this.brithday = brithday; } @Column(name="_title") public String getTitle() { return title; } public void setTitle(String title) { this.title = title; } }
方法3:
在实体类上的主键get方法上注解@Id
@Id public String getName() { return name; } public void setName(String name) { this.name = name; } @Id @GeneratedValue(strategy=GenerationType.AUTO) public int getId() { return id; } public void setId(int id) { this.id = id; }
在实体类的头来标示主键的类:
@IdClass(value=teacherPK.class) public class Teacher {......
最常用的是第1和第二种方法。
转载请注明出处:http://blog.csdn.net/acmman