理解JPA注解@GeneratedValue的使用方法

时间:2022-09-04 21:50:17

一、jpa通用策略生成器

通过annotation来映射hibernate实体的,基于annotation的hibernate主键标识为@id,

其生成规则由@generatedvalue设定的.这里的@id和@generatedvalue都是jpa的标准用法,

jpa提供四种标准用法,由@generatedvalue的源代码可以明显看出.

?
1
2
3
4
5
6
target({method,field}) 
  @retention(runtime) 
  public @interface generatedvalue{ 
    generationtype strategy() default auto; 
    string generator() default ""
  }

其中generationtype:

?
1
2
3
4
5
6
public enum generationtype{ 
  table, 
  sequence, 
  identity, 
  auto 
}

jpa提供的四种标准用法为table,sequence,identity,auto.

  • table:使用一个特定的数据库表格来保存主键。
  • sequence:根据底层数据库的序列来生成主键,条件是数据库支持序列。
  • identity:主键由数据库自动生成(主要是自动增长型)
  • auto:主键由程序控制。

table比较复杂,这里不讲解。分别介绍其他三个:

1.sequence

实体类中的注解

?
1
2
3
@id
@generatedvalue(strategy =generationtype.sequence,generator="aaa")
@sequencegenerator(name="aaa", sequencename="seq_payment")

@sequencegenerator定义

?
1
2
3
4
5
6
7
8
@target({type, method, field}) 
@retention(runtime)
public @interface sequencegenerator {
 string name();
 string sequencename() default "";
 int initialvalue() default 0;
 int allocationsize() default 50;
}
  • name属性表示该表主键生成策略的名称,它被引用在@generatedvalue中设置的“generator”值中。
  • sequencename属性表示生成策略用到的数据库序列名称。
  • initialvalue表示主键初识值,默认为0。
  • allocationsize表示每次主键值增加的大小,例如设置成1,则表示每次创建新记录后自动加1,默认为50。

2.identity

主键则由数据库自动维护,使用起来很简单

?
1
2
@id
@generatedvalue(strategy = generationtype.identity)

3、auto

默认的配置。如果不指定主键生成策略,默认为auto。

?
1
2
@id
@generatedvalue(strategy = generationtype.auto)

二、hibernate主键策略生成器

hibernate提供多种主键生成策略,有点是类似于jpa,有的是hibernate特有,下面列出几个hibernate比较常用的生成策略:

  • native: 对于 oracle 采用 sequence 方式,对于mysql 和 sql server 采用identity(自增主键生成机制),native就是将主键的生成工作交由数据库完成,hibernate不管
  • uuid: 采用128位的uuid算法生成主键,uuid被编码为一个32位16进制数字的字符串。占用空间大(字符串类型)。
  • assigned: 在插入数据的时候主键由程序处理(即程序员手动指定),这是 <generator>元素没有指定时的默认生成策略。等同于jpa中的auto。
  • identity: 使用sql server 和 mysql 的自增字段,这个方法不能放到 oracle 中,oracle 不支持自增字段,要设定sequence(mysql 和 sql server 中很常用)。 等同于jpa中的indentity。
  • increment: 插入数据的时候hibernate会给主键添加一个自增的主键,但是一个hibernate实例就维护一个计数器,所以在多个实例运行的时候不能使用这个方法。

hibernate提供了多种生成器供选择,基于annotation的方式通过@genericgenerator实现.

hibernate每种主键生成策略提供接口org.hibernate.id.identifiergenerator的实现类,如果要实现自定义的主键生成策略也必须实现此接口.

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public interface identifiergenerator {
  /**
   * the configuration parameter holding the entity name
   */
  public static final string entity_name = "entity_name";
 
 /**
  * generate a new identifier.
  * @param session
  * @param object the entity or toplevel collection for which the id is being generated
  *
  * @return a new identifier
  * @throws hibernateexception
  */
 public serializable generate(sessionimplementor session, object object) 
  throws hibernateexception;
}

identifiergenerator提供一generate方法,generate方法返回产生的主键.

三、@genericgenerator

自定义主键生成策略,由@genericgenerator实现。

hibernate在jpa的基础上进行了扩展,可以用一下方式引入hibernate独有的主键生成策略,就是通过@genericgenerator加入的。

比如说,jpa标准用法

?
1
2
@id
@generatedvalue(generationtype.auto)

就可以用hibernate特有以下用法来代替:

?
1
2
3
@id
@generatedvalue(generator = "paymentablegenerator"
@genericgenerator(name = "paymentablegenerator", strategy = "assigned")

@genericgenerator的定义:

?
1
2
@target({package, type, method, field})
@retention(runtime)

源代码如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public @interface genericgenerator {
 /**
 * unique generator name
 */
 string name();
 /**
 * generator strategy either a predefined hibernate
 * strategy or a fully qualified class name.
 */
 string strategy();
 /**
 * optional generator parameters
 */
 parameter[] parameters() default {};
}
  • name属性指定生成器名称。
  • strategy属性指定具体生成器的类名。
  • parameters得到strategy指定的具体生成器所用到的参数。

对于这些hibernate主键生成策略和各自的具体生成器之间的关系,在org.hibernate.id.identifiergeneratorfactory中指定了,

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
static {
 generators.put("uuid", uuidhexgenerator.class);
 generators.put("hilo", tablehilogenerator.class);
 generators.put("assigned", assigned.class);
 generators.put("identity", identitygenerator.class);
 generators.put("select", selectgenerator.class);
 generators.put("sequence", sequencegenerator.class);
 generators.put("seqhilo", sequencehilogenerator.class);
 generators.put("increment", incrementgenerator.class);
 generators.put("foreign", foreigngenerator.class);
 generators.put("guid", guidgenerator.class);
 generators.put("uuid.hex", uuidhexgenerator.class); //uuid.hex is deprecated
 generators.put("sequence-identity", sequenceidentitygenerator.class);
}

上面十二种策略,加上native,hibernate一共默认支持十三种生成策略。使用hibernate注解示例如下:

?
1
2
3
@id
@generatedvalue(generator = "idgenerator")
@genericgenerator(name = "idgenerator", strategy = "identity")

这种完全类似于:

?
1
2
@id
@generatedvalue(strategy=generationtype.identity)

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。

原文链接:https://www.jianshu.com/p/62d12ef20d6b