基于按annotation的hibernate主键生成策略

http://zwtlong.iteye.com/blog/monthblog/2010-12?show_full=true


自定义主键生成策略,由@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、SQL Server、MySQL 中的主键生成机制。

“sequence”
采用数据库提供的 sequence 机制生成主键。如 Oralce 中的Sequence。

“native”
由 Hibernate 根据使用的数据库自行判断采用 identity、hilo、sequence 其中一种作为主键生成方式。

“uuid.hex”
由 Hibernate 基于128 位 UUID 算法 生成16 进制数值(编码后以长度32 的字符串表示)作为主键。

“uuid.string”
与uuid.hex 类似,只是生成的主键未进行编码(长度16),不能应用在 PostgreSQL 数据库中。

“foreign”
使用另外一个相关联的对象的标识符作为主键。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值