Hibernate 提供了三种方式将POJO变成PO类
- 使用持久化注解(以JPA标准注解为主,如果有一些特殊要求,则依然需要使用Hibernate本身提供的注解)
- 使用JPA2提供的XML配置描述文件,这种方式可以让Hibernate的PO类与JPA实体类兼容
- 使用Hibernate传统的XML映射文件(hbm文件)。
1 Hibernate映射主键、属性
(1)使用注解的方式
hibernate主键生成
通常情况下如果实体类的标识属性是基本数据类型、基本类型的包装类、String、Date等类型,可以简单的使用@ID修饰该实体的属性即可。
如果希望Hibernate为逻辑主键自动生成主键值,可以使用@GeneratedValue来修饰实体的标识属性。
| 属性 | 是否必须 | 说明 |
| strategy | 否 | 指定Hibernate对该主键列使用怎么样的主键生成策略
|
| generator | 否 | 当使用GenerationType.SEQUENCE,GenerationType.TABLE主键生成策略时,该属性引用@SequenceGenerator,@TableGenerator所定义的生成器的名称 |
Hibernate主键生成策略
JPA注解只支持AUTO、Identity、SEQUENCE、TABLE这4中生成策略,如果希望使用Hibernate提供的主键生成策略,就需要使用Hibernate本身的@GenericGenerator注解,该注解用于定义生成器。包括name和strategy两个属性。
stratety属性可指定的值:
- native: 对于 oracle 采用 Sequence 方式,对于MySQL 和 SQL Server 采用identity(自增主键生成机制),native就是将主键的生成工作交由数据库完成,hibernate不管(很常用)。
- uuid: 采用128位的uuid算法生成主键,uuid被编码为一个32位16进制数字的字符串。占用空间大(字符串类型)。
- hilo: 使用hilo生成策略,要在数据库中建立一张额外的表,默认表名为hibernate_unique_key,默认字段为integer类型,名称是next_hi(比较少用)。
- assigned: 在插入数据的时候主键由程序处理(很常用),这是 <generator>元素没有指定时的默认生成策略。等同于JPA中的AUTO。
- identity: 使用SQL Server 和 MySQL 的自增字段,这个方法不能放到 Oracle 中,Oracle 不支持自增字段,要设定sequence(MySQL 和 SQL Server 中很常用)。
- select: 使用触发器生成主键(主要用于早期的数据库主键生成机制,少用)。
- sequence: 调用底层数据库的序列来生成主键,要设定序列名,不然hibernate无法找到。
- seqhilo: 通过hilo算法实现,但是主键历史保存在Sequence中,适用于支持 Sequence 的数据库,如 Oracle(比较少用)
- increment: 插入数据的时候hibernate会给主键添加一个自增的主键,但是一个hibernate实例就维护一个计数器,所以在多个实例运行的时候不能使用这个方法。
- foreign: 使用另外一个相关联的对象的主键。通常和<one-to-one>联合起来使用。
- guid: 采用数据库底层的guid算法机制,对应MYSQL的uuid()函数,SQL Server的newid()函数,ORACLE的rawtohex(sys_guid())函数等。
- uuid.hex: 看uuid,建议用uuid替换。
- sequence-identity: sequence策略的扩展,采用立即检索策略来获取sequence值,需要JDBC3.0和JDK4以上(含1.4)版本
// 消息类的标识属性
@Id @Column(name="news_id")
// 使用@GenericGenerator定义主键生成器。
// 该主键生成器名为fk_hilo,使用Hibernate的hilo策略,
@GenericGenerator(name="fk_hilo" , strategy="hilo")
// 指定使用fk_hilo主键生成器
@GeneratedValue(generator="fk_hilo")
private Integer id;(2) 使用XML配置文件的方式
<hibernate-mapping>
<class name="com.songxu.modle.Person" table="person">
<id column="id" name="id" type="int">
<generator class="increment"/>
</id>
<!-- column同名时可以省略 -->
<property column="name" generated="never" lazy="false" name="name" type="string"/>
<property column="age" generated="never" lazy="false" name="age" type="int"/>
<property column="registertime" generated="never" lazy="false"
name="time" type="date"/>
</class>
</hibernate-mapping><id column="id" name="id" type="string">
<generator class="uuid"></generator>2 Hibernate 映射集合属性
集合属性大致分为两种:一种是单纯的属性集合,例如List、Set或数组等集合属性;另一种是Map结构的集合属性,每个属性值都有对应的Key映射。
不管哪种类型的集合属性,都统一用@ElementCollection 注解进行映射。
在Java的所有集合中,只有Set集合是无序的,即没有显示的索引值。List、数组使用整数作为集合元素的索引值,而Map则使用key作为集合的索引,因此,如果要映射带索引的集合,即需要为集合袁术所在的数据表指定一个索引列,用于保存数组索引、List索引或者是Map集合的Key索引。
集合类型大致可分为如下几种情况:
- 集合袁术是基本类型及其包装类、字符串类型和日期类型:此时使用@ElementCollection映射集合属性,并使用普通的@Column映射集合元素对应的列。
- 集合元素是组件:此时使用@ElementCollection 映射集合属性,然后使用@Embeddable修饰非持久化实体的复合类。
- 集合元素是关联的持久化实体:此时已经不再是集合属性,应该使用@OneToMany或@ManyToMany进行关联映射
2.1 List集合属性
List是有序集合,因此持久化到数据库时也必须增加一列来表示集合的次序。
以下Person类 添加了集合属性list用于保存学校的名称
@Id @Column(name="perosn_id")
@GeneratedValue(strategy=GenerationType.IDENTITY)
// 标识属性
private Integer id;
private String name;
private int age;
// 集合属性,保留该对象关联的学校
@ElementCollection(targetClass=String.class)
// 映射保存集合属性的表
@CollectionTable(name="school_inf", // 指定表名为school_inf
joinColumns=@JoinColumn(name="person_id" , nullable=false))
// 指定保存集合元素的列为 school_name
@Column(name="school_name")
// 映射集合元素索引的列
@OrderColumn(name="list_order")
private List<String> schools
= new ArrayList<>();
生成的表后,List集合属性的表总是以外键列和元素索引列作为联合主键2.2 Set集合属性
Set是无序集合,因此无需使用@OrderColumn注解映射集合元素的索引列。
// 集合属性,保留该对象关联的学校
@ElementCollection(targetClass=String.class)
// 映射保存集合属性的表
@CollectionTable(name="school_inf", // 指定表名为school_inf
joinColumns=@JoinColumn(name="person_id" , nullable=false))
// 指定保存集合元素的列为 school_name,nullable=false增加非空约束
@Column(name="school_name" , nullable=false)
private Set<String> schools
= new HashSet<>();
2.3Map属性
Map属性需要使用@MapKeyColumn映射保存Map Key的数据列
// 集合属性,保留该对象关联的考试成绩
@ElementCollection(targetClass=Float.class)
// 映射保存集合属性的表
@CollectionTable(name="score_inf", // 指定表名为score_inf
joinColumns=@JoinColumn(name="person_id" , nullable=false))
@MapKeyColumn(name="subject_name")
// 指定Map key的类型为String类型
@MapKeyClass(String.class)
// 映射保存Map value的数据列
@Column(name="mark")
private Map<String , Float> scores
= new HashMap<>();虽然程序定义了Person类使用了泛型来显示Map集合的Key、Value的类型,但程序中依然通过注解强制执行Map Key MapValue 的类型,这样可以避免Hibernate通过反射去获取,从而提升了程序性能。
如果注解与Person类定义的泛型指定的类型不一致时,Hibernate将通过注解类型进行数据库表的生成工作。当插入数据时,会抛出数据类型不匹配的异常。
生成的保存Map数据的数据表将使用外键列和Map中的 key作为联合主键
2.4 组件
组件属性是指持久化类的属性不是基本数据类型,也不是字符串、日期等标量类型,而是一个复合类型的。
下面是一个组件的示例。定义了一个Name类型,使用@Embeddable注解,该注解与@Entity类似。该类包含一个owner属性,该属性指向包含该Name属性的实体。为了告诉Hibernate这个owner属性不是普通属性,而是包含Name组件的Person实体,可使用@Parent注解修饰该属性。
(1) 定义单独的组件类
@Embeddable
public class Name
{
// 定义first成员变量
@Column(name="person_firstname")
private String first;
// 定义last成员变量
@Column(name="person_lastname")
private String last;
// 引用拥有该Name的Person对象
@Parent // ①
private Person owner;
// 无参数的构造器
public Name()
{
}
// 初始化全部成员变量的构造器
public Name(String first , String last)
{
this.first = first;
this.last = last;
}
// first的setter和getter方法
public void setFirst(String first)
{
this.first = first;
}
public String getFirst()
{
return this.first;
}
// last的setter和getter方法
public void setLast(String last)
{
this.last = last;
}
public String getLast()
{
return this.last;
}
// owner的setter和getter方法
public void setOwner(Person owner)
{
this.owner = owner;
}
public Person getOwner()
{
return this.owner;
}
}@Id @Column(name="person_id")
@GeneratedValue(strategy=GenerationType.IDENTITY)
private Integer id;
private int age;
// 组件属性name
private Name name;(2)持久化类内部定义组件
这种方式无需使用@Embeddable注解修饰,而是直接在持久化类中使用@Embedded注解修饰组件属性
public class Name
{
// 定义first成员变量
private String first;
// 定义last成员变量
private String last;
// 引用拥有该Name的Person对象
@Parent
private Person owner;
// 无参数的构造器
public Name()
{
}
// 初始化全部成员变量的构造器
public Name(String first , String last)
{
this.first = first;
this.last = last;
}
// first的setter和getter方法
public void setFirst(String first)
{
this.first = first;
}
public String getFirst()
{
return this.first;
}
// last的setter和getter方法
public void setLast(String last)
{
this.last = last;
}
public String getLast()
{
return this.last;
}
// owner的setter和getter方法
public void setOwner(Person owner)
{
this.owner = owner;
}
public Person getOwner()
{
return this.owner;
}
}@Embedded
@AttributeOverrides({
@AttributeOverride(name="first", column = @Column(name="person_firstname")),
@AttributeOverride(name="last", column = @Column(name="person_lastname"))
})
private Name name;
2.5 组件属性为集合
如果组件包含了 list map set集合,可以直接在组件类中使用@ElementCollection修饰集合属性,并使用@CollectionTable指定保存集合属性的属性表。
@Embeddable
public class Name
{
// 定义first成员变量
@Column(name="person_firstname")
private String first;
// 定义last成员变量
@Column(name="person_lastname")
private String last;
// 引用拥有该Name的Person对象
@Parent
private Person owner;
// 集合属性,保留该对象关联的考试成绩
@ElementCollection(targetClass=Integer.class)
@CollectionTable(name="power_inf",
joinColumns=@JoinColumn(name="person_name_id" , nullable=false))
@MapKeyColumn(name="name_aspect")
@Column(name="name_power" , nullable=false)
@MapKeyClass(String.class)
private Map<String , Integer> power
= new HashMap<>();
// 无参数的构造器
public Name()
{
}
}
2.6 组件作为复合主键
使用组件作为复合主键,也就是使用组件作为持久化类的标识符,则该组件类必须满足以下要求。
- 有无参数的构造器。
- 必修实现java.io.Serializabel接口。(在Hibernate4中,不是必须的)
- 建议正确地重写equals()和hashCode()方法,也就是根据组件类的关键属性来区分组件对象。(因为组件是唯一标识符,必须重写这两个方法。)
public class Name
implements java.io.Serializable
{
// 定义first成员变量
private String first;
// 定义last成员变量
private String last;
// 无参数的构造器
public Name()
{
}
// 初始化全部成员变量的构造器
public Name(String first , String last)
{
this.first = first;
this.last = last;
}
// first的setter和getter方法
public void setFirst(String first)
{
this.first = first;
}
public String getFirst()
{
return this.first;
}
// last的setter和getter方法
public void setLast(String last)
{
this.last = last;
}
public String getLast()
{
return this.last;
}
// 重写equals()方法,根据first、last进行判断
public boolean equals(Object obj)
{
if (this == obj)
{
return true;
}
if (obj != null && obj.getClass() == Name.class)
{
Name target = (Name)obj;
return target.getFirst().equals(getFirst())
&& target.getLast().equals(getLast());
}
return false;
}
// 重写hashCode()方法,根据first、last计算hashCode值
public int hashCode()
{
return getFirst().hashCode() * 31
+ getLast().hashCode();
}
}// 以Name组件作为标识属性
@EmbeddedId
@AttributeOverrides({
// 指定
@AttributeOverride(name="first",
column = @Column(name="person_firstname")),
@AttributeOverride(name="last",
column = @Column(name="person_lastname"))
})
private Name name;2.7多列作为联合主键
Hibernate 还提供了另一种联合主键支持,如果需要直接将持久化类的多列映射成联合主键,则该持久化类必须满足如下条件。
- 有无参数的构造器。
- 必修实现java.io.Serializabel接口。(在Hibernate4中,不是必须的)
- 建议根据联合主键列所映射的属性重写equals()和hashCode()方法,也就是根据组件类的关键属性来区分组件对象。(因为组件是唯一标识符,必须重写这两个方法。)
@Entity
@Table(name="person_inf")
public class Person
implements java.io.Serializable
{
// 定义first属性,作为标识属性的成员
@Id
private String first;
// 定义last属性,作为标识属性的成员
@Id
private String last;
private int age;
// first的setter和getter方法
public void setFirst(String first)
{
this.first = first;
}
public String getFirst()
{
return this.first;
}
// last的setter和getter方法
public void setLast(String last)
{
this.last = last;
}
public String getLast()
{
return this.last;
}
// age的setter和getter方法
public void setAge(int age)
{
this.age = age;
}
public int getAge()
{
return this.age;
}
// 重写equals()方法,根据first、last进行判断
public boolean equals(Object obj)
{
if (this == obj)
{
return true;
}
if (obj != null && obj.getClass() == Person.class)
{
Person target = (Person)obj;
return target.getFirst().equals(getFirst())
&& target.getLast().equals(getLast());
}
return false;
}
// 重写hashCode()方法,根据first、last计算hashCode值
public int hashCode()
{
return getFirst().hashCode() * 31
+ getLast().hashCode();
}
}
本文详细介绍了Hibernate映射主键、属性的方式,包括使用注解和XML配置文件。主键生成策略有多种,如native、uuid、hilo等。在映射集合属性时,涉及List、Set、Map以及组件的不同处理。组件可以是持久化类内部定义或单独的类,甚至可以作为复合主键。最后,文章讨论了多列作为联合主键的情况。
1万+

被折叠的 条评论
为什么被折叠?



