基于按annotation的hibernate主键生成策略
自定义主键生成策略,由@GenericGenerator实现。
hibernate在JPA的基础上进行了扩展,可以用一下方式引入hibernate独有的主键生成策略,就是通过@GenericGenerator加入的。
比如说,JPA标准用法
Java代码
@Id
@GeneratedValue(GenerationType.AUTO)
就可以用hibernate特有以下用法来实现
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="paymentableGenerator",strategy="assigned")
@GenericGenerator的定义:
Java代码
@Target({PACKAGE,TYPE,METHOD,FIELD})
@Retention(RUNTIME)
public@interfaceGenericGenerator{
Stringname();
Stringstrategy();
Parameter[]parameters()default{};
}
name属性指定生成器名称。
strategy属性指定具体生成器的类名。
parameters得到strategy指定的具体生成器所用到的参数。
对于这些hibernate主键生成策略和各自的具体生成器之间的关系,在org.hibernate.id.IdentifierGeneratorFactory中指定了,
Java代码
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.hexisdeprecated
GENERATORS.put("sequence-identity",SequenceIdentityGenerator.class);
}
上面十二种策略,加上native,hibernate一共默认支持十三种生成策略。
1、native
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="paymentableGenerator",strategy="native")
2、uuid
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="paymentableGenerator",strategy="uuid")
3、hilo
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="paymentableGenerator",strategy="hilo")
4、assigned
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="paymentableGenerator",strategy="assigned")
5、identity
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="paymentableGenerator",strategy="identity")
6、select
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="select",strategy="select",
parameters={@Parameter(name="key",value="idstoerung")})
7、sequence
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="paymentableGenerator",strategy="sequence",
parameters={@Parameter(name="sequence",value="seq_payablemoney")})
8、seqhilo
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="paymentableGenerator",strategy="seqhilo",
parameters={@Parameter(name="max_lo",value="5")})
9、increment
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="paymentableGenerator",strategy="increment")
10、foreign
Java代码
@GeneratedValue(generator="idGenerator")
@GenericGenerator(name="idGenerator",strategy="foreign",
parameters={@Parameter(name="property",value="employee")})
注意:直接使用@PrimaryKeyJoinColumn报错(?)
Java代码
@OneToOne(cascade=CascadeType.ALL)
@PrimaryKeyJoinColumn
例如
Java代码
@Entity
publicclassEmployee{
@IdIntegerid;
@OneToOne@PrimaryKeyJoinColumn
EmployeeInfoinfo;
}
应该为
Java代码
@Entity
publicclassEmployee{
@Id
@GeneratedValue(generator="idGenerator")
@GenericGenerator(name="idGenerator",strategy="foreign",
parameters={@Parameter(name="property",value="info")})
Integerid;
@OneToOne
EmployeeInfoinfo;
}
11、guid
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="paymentableGenerator",strategy="guid")
12、uuid.hex
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="paymentableGenerator",strategy="uuid.hex")
13、sequence-identity
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="paymentableGenerator",strategy="sequence-identity",
parameters={@Parameter(name="sequence",value="seq_payablemoney")})
四、通过@GenericGenerator自定义主键生成策略
如果实际应用中,主键策略为程序指定了就用程序指定的主键(assigned),没有指定就从sequence中取。
明显上面所讨论的策略都不满足,只好自己扩展了,集成assigned和sequence两种策略。
Java代码
publicclassAssignedSequenceGeneratorextendsSequenceGeneratorimplements
PersistentIdentifierGenerator,Configurable{
privateStringentityName;
publicvoidconfigure(Typetype,Propertiesparams,Dialectdialect)throwsMappingException{
entityName=params.getProperty(ENTITY_NAME);
if(entityName==null){
thrownewMappingException("noentityname");
}
super.configure(type,params,dialect);
}
publicSerializablegenerate(SessionImplementorsession,Objectobj)
throwsHibernateException{
Serializableid=session.getEntityPersister(entityName,obj)
.getIdentifier(obj,session.getEntityMode());
if(id==null){
id=super.generate(session,obj);
}
returnid;
}
}
实际应用中,定义同sequence。
Java代码
@GeneratedValue(generator="paymentableGenerator")
@GenericGenerator(name="paymentableGenerator",strategy="AssignedSequenceGenerator",
parameters={@Parameter(name="sequence",value="seq_payablemoney")})
值得注意的是,定义的这种策略,就像打开了潘多拉魔盒,非常不可控。正常情况下,不建议这么做。策略解释
“assigned”
主键由外部程序负责生成,在save()之前指定一个。
“hilo”
通过hi/lo算法实现的主键生成机制,需要额外的数据库表或字段提供高位值来源
“seqhilo”
与hilo类似,通过hi/lo算法实现的主键生成机制,需要数据库中的Sequence,适用于支持Sequence的数据库,如Oracle。
“increment”
主键按数值顺序递增。此方式的实现机制为在当前应用实例中维持一个变量,以保存着当前的最大值,之后每次需要生成主键的时候将此值加1作为主键。这种方式可能产生的问题是:不能在集群下使用。
“identity”
采用数据库提供的主键生成机制。如DB2、SQLServer、MySQL中的主键生成机制。
“sequence”
采用数据库提供的sequence机制生成主键。如Oralce中的Sequence。
“native”
由Hibernate根据使用的数据库自行判断采用identity、hilo、sequence其中一种作为主键生成方式。
“uuid.hex”
由Hibernate基于128位UUID算法生成16进制数值(编码后以长度32的字符串表示)作为主键。
“uuid.string”
与uuid.hex类似,只是生成的主键未进行编码(长度16),不能应用在PostgreSQL数据库中。
“foreign” 使用另外一个相关联的对象的标识符作为主键。