当前位置: 移动技术网 > IT编程>开发语言>Java > Java探索之Hibernate主键生成策略详细介绍

Java探索之Hibernate主键生成策略详细介绍

2019年07月19日  | 移动技术网IT编程  | 我要评论

1.increment

     由hibernate从数据库中去除主键的最大值(每个session只取一次),以该值为基础,每次增量为1,在内存中生成主键,不依赖于底层的数据库,因此可以跨数据库。

<id name="id" column="id">
<generator class="increment" />
</id>

hibernate调用org.hibernate.id.incrementgenerator类里面的generate()方法,使用select max(idcolumnname) from tablename语句获取主键最大值。该方法被声明成了synchronized,所以在一个独立的java虚拟机内部是没有问题的,然而,在多个jvm同时并发访问数据库select max时就可能取出相同的值,再insert就会发生dumplicate entry的错误。所以只能有一个hibernate应用进程访问数据库,否则就可能产生主键冲突,所以不适合多进程并发更新数据库,适合单一进程访问数据库,不能用于群集环境。

特点:跨数据库,不适合多进程并发更新数据库,适合单一进程访问数据库,不能用于群集环境。

2.sequence

采用数据库提供的sequence机制生成主键,需要数据库支持sequence。如oralce、db、sap db、postgersql、mckoi中的sequence。mysql这种不支持sequence的数据库则不行(可以使用identity)。

<generator class="sequence">
<param name="sequence">hibernate_id</param>
</generator>

<param name="sequence">hibernate_id</param> 指定sequence的名称

hibernate生成主键时,查找sequence并赋给主键值,主键值由数据库生成,hibernate不负责维护,使用时必须先创建一个sequence,如果不指定sequence名称,则使用hibernate默认的sequence,名称为hibernate_sequence,前提要在数据库中创建该sequence。

特点:只能在支持序列的数据库中使用,如oracle。

3.identity

identity由底层数据库生成标识符。identity是由数据库自己生成的,但这个主键必须设置为自增长,使用identity的前提条件是底层数据库支持自动增长字段类型,如db2、sql server、mysql、sybase和hypersonicsql等,oracle这类没有自增字段的则不支持。

<id name="id" column="id">
<generator class="identity" />
</id>

例:如果使用mysql数据库,则主键字段必须设置成auto_increment。
id int(11) primary key auto_increment

特点:只能用在支持自动增长的字段数据库中使用,如mysql。

4.native

    native由hibernate根据使用的数据库自行判断采用identity、hilo、sequence其中一种作为主键生成方式,灵活性很强。如果能支持identity则使用identity,如果支持sequence则使用sequence。

<id name="id" column="id">
<generator class="native" />
</id>

例如mysql使用identity,oracle使用sequence
注意:如果hibernate自动选择sequence或者hilo,则所有的表的主键都会从hibernate默认的sequence或hilo表中取。并且,有的数据库对于默认情况主键生成测试的支持,效率并不是很高。
使用sequence或hilo时,可以加入参数,指定sequence名称或hi值表名称等,如
<param name="sequence">hibernate_id</param>

    特点:根据数据库自动选择,项目中如果用到多个数据库时,可以使用这种方式,使用时需要设置表的自增字段或建立序列,建立表等。

5.uuid

    uuid:universally unique identifier,是指在一台机器上生成的数字,它保证对在同一时空中的所有机器都是唯一的。按照开放软件基金会(osf)制定的标准计算,用到了以太网卡地址、纳秒级时间、芯片id码和许多可能的数字,标准的uuid格式为:

xxxxxxxx-xxxx-xxxx-xxxxxx-xxxxxxxxxx (8-4-4-4-12)

其中每个 x 是 0-9 或 a-f 范围内的一个十六进制的数字。

<id name="id" column="id">
<generator class="uuid" />
</id>

hibernate在保存对象时,生成一个uuid字符串作为主键,保证了唯一性,但其并无任何业务逻辑意义,只能作为主键,唯一缺点长度较大,32位(hibernate将uuid中间的“-”删除了)的字符串,占用存储空间大,但是有两个很重要的优点,hibernate在维护主键时,不用去数据库查询,从而提高效率,而且它是跨数据库的,以后切换数据库极其方便。

    特点:uuid长度大,占用空间大,跨数据库,不用访问数据库就生成主键值,所以效率高且能保证唯一性,移植非常方便,推荐使用。

6.guid

    guid:globally unique identifier全球唯一标识符,也称作 uuid,是一个128位长的数字,用16进制表示。算法的核心思想是结合机器的网卡、当地时间、一个随即数来生成guid。从理论上讲,如果一台机器每秒产生10000000个guid,则可以保证(概率意义上)3240年不重复。

<id name="id" column="id">
<generator class="guid" />
</id>

hibernate在维护主键时,先查询数据库,获得一个uuid字符串,该字符串就是主键值,该值唯一,缺点长度较大,支持数据库有限,优点同uuid,跨数据库,但是仍然需要访问数据库。

注意:长度因数据库不同而不同

mysql中使用select uuid()语句获得的为36位(包含标准格式的“-”)
oracle中,使用select rawtohex(sys_guid()) from dual语句获得的为32位(不包含“-”) 

    特点:需要数据库支持查询uuid,生成时需要查询数据库,效率没有uuid高,推荐使用uuid。

7.foreign

    使用另外一个相关联的对象的主键作为该对象主键。主要用于一对一关系中。

<id name="id" column="id">
<generator class="foreign">
<param name="property">user</param>
</generator>
</id>
<one-to-one name="user" class="domain.user" constrained="true" />

该例使用domain.user的主键作为本类映射的主键。

特点:很少使用,大多用在一对一关系中。

总结

以上就是本文关于java探索之hibernate主键生成策略详细介绍的全部内容,希望对大家有所帮助。感兴趣的朋友可以继续参阅本站:hibernate实现悲观锁和乐观锁代码介绍详细解读hibernate的缓存机制等,有什么问题可以随时留言,欢迎交流讨论。感谢朋友们对本站的支持!

如对本文有疑问, 点击进行留言回复!!

相关文章:

验证码:
移动技术网