2 Hibernate映射 属性 主键 集合 组件 联合主键

Hibernate 提供了三种方式将POJO变成PO类

  • 使用持久化注解(以JPA标准注解为主,如果有一些特殊要求,则依然需要使用Hibernate本身提供的注解)
  • 使用JPA2提供的XML配置描述文件,这种方式可以让Hibernate的PO类与JPA实体类兼容
  • 使用Hibernate传统的XML映射文件(hbm文件)。

1 Hibernate映射主键、属性


(1)使用注解的方式

hibernate主键生成

通常情况下如果实体类的标识属性是基本数据类型、基本类型的包装类、String、Date等类型,可以简单的使用@ID修饰该实体的属性即可。

如果希望Hibernate为逻辑主键自动生成主键值,可以使用@GeneratedValue来修饰实体的标识属性。


GeneratedValue支持的属性
属性 是否必须 说明
  strategy 指定Hibernate对该主键列使用怎么样的主键生成策略
  • GenerationType.Auto:hibernate自动选择最社和底层数据库的主键生成策略,这是默认值
  • GenerationType.IDENTITY:对于MySQL、SQL Sever这样的数据库,选择自增长的主键生成策略
  • GenerationType.SEQUENCE:对于Oracle这样的数据库,选择使用基于sequence的主键生成策略。应与@SequenceGenerator一起使用
  • GenerationType.TABLE:使用辅助表来生成主键。应与@TableGenerator一起使用
generator 当使用GenerationType.SEQUENCEGenerationType.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 映射集合属性

Hibernate映射的集合对应Java中的set list map对象,在这里面仅介绍以注解方式配置。
集合属性大致分为两种:一种是单纯的属性集合,例如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();
	}
}
























你可能感兴趣的:(2 Hibernate映射 属性 主键 集合 组件 联合主键)