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文件中关于标识符属性的设置信息如下:
选择标识符属性生成策略
在选择Hibernate提供的标识符属性生成策略时,要具体问题具体分析,如果应用系统不需要分布式部署,在数据库支持的情况下使用sequence、identity、hilo、seqhilo和uuid生成策略都是不错选择;如果应用需要使用多个数据库或者进行分布式的部署,则uuid生成策略是最佳的选择。还有一种情况是使用Hibernate改造遗留系统,可能需要人工采用一定的规则为标识符属性赋值,这时使用assigned生成策略比较合适。