标识符属性的生成策略
Hibernate映射文件(*.hbm.xml)中的<id>元素定义持久化类的标识符属性名、类型和与数据库表中字段的映射,其<generator>子元素则用来设置当前持久化类的标识符属性的生成策略。一个持久化对象对应数据库表中的一条记录,为了更好地持久化表中的每一条记录,需要通过<generator>元素的class属性设置采用的标识符属性生成策略,如下面的代码所示:
<id name="id" type="java.lang.Integer" column="ID">
<generator class="increment">
</generator>
</id>
<param>元素是<generator>元素的可选子元素,用来传递标识符属性生成时可能需要的一个或多个参数。
在Hibernate中内置了多种标识符属性生成策略,如果不能满足需要,还可以通过实现org.hibernate.id.IdentifierGenerator接口来定制标识符属性生成策略。
increment生成策略
当Hibernate准备在数据库表中插入一条新记录时,首先从数据库表中获取当前主键字段的最大值,然后在最大值基础上加1,作为当前持久化对象的标识符属性值。这种策略即increment生成策略,用其生成的标识符属性的类型可以是long、short、int及其封装类的类型。这种主键生成策略适合单服务器的Hibernate应用,同时数据库也只被这个Hibernate应用所独享;否则不能保证生成的标识符属性值的唯一性,即这种标识符属性生成策略并不适合非独享数据库或者分布式的Hibernate应用。使用这种标识符属性生成策略的示例配置信息如下:
<id name="id" type="java.lang.Integer" column="ID">
<generator class="increment">
</generator>
</id>
identity生成策略
在MS SQL Server、MySQL和DB2等数据库中可以设置表中某一个字段的数值自动增长,identity生成策略通过这种方式为当前记录获取主键值的同时为持久化对象赋予标识符属性值。该生成策略生成的标识符属性的类型可以是long、short、int及其封装类的类型。使用这种标识符属性生成策略的示例配置信息如下:
<id name="id" type="java.lang.Integer" column="ID">
<generator class="identity">
</generator>
</id>
sequence生成策略
在Oracle、DB2和PostgreSQL等数据库中创建一个序列(sequence),然后Hibernate通过该序列为当前记录获取主键值,进而为持久化对象赋予标识符属性值。此即sequence生成策略,用其生成的标识符属性的类型可以是long、short、int及其封装类的类型。使用这种标识符属性生成策略的示例配置信息如下:
<id name="id" type="java.lang.Integer" column="ID">
<generator class="sequence">
<param name="sequence">gb_seq</param>
</generator>
</id>
第3行指定Hibernate使用的序列名,该行为可选。如果未指定序列名,则Hibernate默认使用名为"hibernate_sequence"的序列。
hilo生成策略
hilo生成策略采用一种称为"高/低位"(hi/lo)的高效算法产生标识符属性值,所产生的标识符属性值为long、short、int及其封装类的类型。该算法使用一个高位值(hi)和一个低位值(lo),然后计算其值,运算结果作为标识符属性值。使用这种标识符属性生成策略需要在数据库中建立一个表和一个字段(默认表名为"hibernate_unique_key",字段名为"next_hi",该字段要有数值)作为高位值的来源。使用这种标识符属性生成策略的示例配置信息如下:
<id name="id" type="java.lang.Integer" column="ID">
<generator class="hilo">
<param name="table">hibernate_key</param>
<param name="column">next_hivalue</param>
</generator>
</id>
这种标识符属性生成策略需要额外的数据库表和字段的支持,可保证为特定数据库生成唯一的标识符属性。并且这种策略与底层使用的数据库无关,可以跨数据库使用。
seqhilo生成策略
seqhilo生成策略也使用高/低位算法,产生的标识符属性值为long、short、int及其封装类的类型。与hilo生成策略不同的是,它使用指定的sequence获取高位值。使用这种标识符属性生成策略的示例配置信息如下:
<id name="id" type="java.lang.Integer" column="ID">
<generator class="seqhilo">
<param name="sequence">hibernate_seq</param>
</generator>
</id>
第3行指定Hibernate使用的序列名,该行为可选。如果未指定序列名,则Hibernate默认使用名为"hibernate_unique_sequence"的序列。
uuid生成策略
uuid生成策略采用128位的UUID算法来生成一个字符串类型的主键值,这个算法使用IP地址、JVM的启动时间(精确到1/4秒)、系统时间和一个计数器值(在当前的JVM中唯一)经过计算来产生标识符属性值,可以用于分布式的Hibernate应用中。产生的标识符属性是一个32位长度的字符串。使用这种生成策略,对应持久化类中标识符属性的类型应该设置为String类型,其示例配置信息如下所示。
<id name="id" type="java.lang.String" column="ID">
<generator class="uuid">
</generator>
</id>
这种标识符属性生成策略生成的数值可以保证多个数据库之间的唯一性,由于该值是32位长的字符串,所以占用的数据库空间较大。推荐在实际开发中使用这种生成策略。
guid生成策略
这种标识符属性生成策略借助MS SQL Server或者MySQL数据库中的GUID字符串产生标识符属性值。如果使用MS SQL Server数据库,表中需要把标识符属性对应的字段类型设置为"uniqueidentifier"。其示例配置信息如下:
<id name="id" type="java.lang.String" column="ID">
<generator class="guid">
</generator>
</id>
native生成策略
由Hibernate根据所使用的数据库支持能力从identity、sequence或者hilo生成策略中选择一种,其示例配置信息如下:
<id name="id" type="java.lang.Integer" column="ID">
<generator class="native">
</generator>
</id>
使用这种标识符属性生成策略可以根据不同的数据库采用不同的生成策略,如Oracle中使用sequence,在MySQL中使用identity便于Hibernate应用在不同的数据库之间移植。
assigned生成策略
assigned生成策略由Hibernate应用自定义标识符属性的数值,即在调用Session对象的save()方法持久化对象时,需要首先为持久化对象的标识符属性赋值。如果<generator>元素没有设置主键生成策略,则默认为assigned生成策略,其示例配置信息如下:
<id name="id" type="java.lang.Integer" column="ID">
<generator class="assigned">
</generator>
</id>
foreign生成策略
foreign生成策略通过关联的持久化对象为当前持久化对象设置标识符属性值,当处理持久化类一对一关联时,一个持久化类的标识符属性值可以参考关联持久化类的标识符属性值获取。例如,User类和Profile类标识符属性都是id,二者是一对一的关联关系。可以设置Profile类的标识符属性值通过User类的标识符属性获取,Profile.hbm.xml文件中关于标识符属性的设置信息如下:
<id name="id" type="java.lang.Integer" column="ID" >
<generator class="foreign">
<param name="property">user</param>
</generator>
</id>
选择标识符属性生成策略
在选择Hibernate提供的标识符属性生成策略时,要具体问题具体分析,如果应用系统不需要分布式部署,在数据库支持的情况下使用sequence、identity、hilo、seqhilo和uuid生成策略都是不错选择;如果应用需要使用多个数据库或者进行分布式的部署,则uuid生成策略是最佳的选择。还有一种情况是使用Hibernate改造遗留系统,可能需要人工采用一定的规则为标识符属性赋值,这时使用assigned生成策略比较合适。