目录
前言
Hibernate 之主键生成策略,其实就是生成数据库主键。非常简单,直接看展例吧
示例
准备
数据库建表
t_hibernate_worker
t_hibernate_student
实体类定义和对应的xml配置
Student.hbm.xml
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="com.cpc.tow.entity.Student" table="t_hibernate_student">
<id name="sid" type="java.lang.Integer" column="sid">
<!--配置主键生成策略例子: (待会会在这里配置)-->
<generator class="xxx" />
</id>
<property name="sname" type="java.lang.String" column="sname">
</property>
</class>
</hibernate-mapping>
Student.java
package com.cpc.tow.entity;
public class Student {
private Integer sid;
private String sname;
public Integer getSid() {
return sid;
}
public void setSid(Integer sid) {
this.sid = sid;
}
public String getSname() {
return sname;
}
public void setSname(String sname) {
this.sname = sname;
}
@Override
public String toString() {
return "Student [sid=" + sid + ", sname=" + sname + "]";
}
}
Worker.hbm.xml
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="com.cpc.tow.entity.Worker" table="t_hibernate_worker">
<id name="wid" type="java.lang.String" column="wid">
<!--配置主键生成策略例子: (待会会在这里配置)-->
<generator class="xxx" />
</id>
<property name="wname" type="java.lang.String" column="wname">
</property>
</class>
</hibernate-mapping>
Worker
package com.cpc.tow.entity;
public class Worker {
private String wid;
private String wname;
public String getWid() {
return wid;
}
public void setWid(String wid) {
this.wid = wid;
}
public String getWname() {
return wname;
}
public void setWname(String wname) {
this.wname = wname;
}
@Override
public String toString() {
return "Worker [wid=" + wid + ", wname=" + wname + "]";
}
}
在Hibernate的配置文件中添加实体映射文件
<!-- 主键生成策略 -->
<mapping resource="com/cpc/tow/entity/Student.hbm.xml" />
<mapping resource="com/cpc/tow/entity/Worker.hbm.xml" />
准备好了后就开始演示了
DemoDao
该类中定义了两个方法
/**
* 添加学生
* @param stu
* @return
*/
public Serializable addStudent(Student stu) {
Session session = SessionFactoryUtils.openSession();
Transaction transaction = session.beginTransaction();
Serializable saveId = session.save(stu);
transaction.commit();
session.close();
return saveId;
}
/**
* 添加工人
* @param stu
* @return
*/
public Serializable addWorker(Worker worker) {
Session session = SessionFactoryUtils.openSession();
Transaction transaction = session.beginTransaction();
Serializable saveId = session.save(worker);
transaction.commit();
session.close();
return saveId;
}
程序员自己控制:assigned
在Student.hbm.xml
中配置:
<id name="sid" type="java.lang.Integer" column="sid">
<generator class="assigned" />
</id>
测试代码
DemoDao dao = new DemoDao();
Student stu = new Student();
//自己指定主键
stu.setSid(123);
stu.setSname("旺旺");
System.out.println(dao.addStudent(stu));
数据库中sid列就是123
啦
注意:sid列段必须手动设置,就是数据库中是自动增长列也不行。以xml配置为标准
数据库控制
identity(标识列/自动增长)
在Student.hbm.xml
中配置:
<id name="sid" type="java.lang.Integer" column="sid">
<generator class="identity" />
</id>
测试代码
DemoDao dao = new DemoDao();
Student stu = new Student();
stu.setSname("旺旺");
System.out.println(dao.addStudent(stu));
数据库中sid列就是使用自动增长列
注意:数据库中sid列段必须为自动增长否则会报错
sequence:内部序列
sequence使用内部序列表来生成主键
mySql数据库中配置
首先我们得在mySql中定义一张hibernate_sequence
表:
并且,在这张表中插入一条初始化数据,为1就行。当然也可以为2。当我们使用内部序列的时候Hibernate就会根据序列表的next_val生成主键,并且修改序列表的next_val(next_val++)方便下一次使用。hibernate_sequence
这张表是默认的也就是不指定序列表时。当然我们也可以指定自己使用的序列表,下面会将,当然指定的表结构和上面操作一样只不过是表名改改。
Oracle数据库中配置
如果你使用的是Oracle数据库那么就是只要创建一个hibernate_sequence
就ok了。
切记数据库的更换一定要修改一下Hibernate的配置文件,除了修改用户密码驱动外还有一个地方要修改,就是Hibernate所采用的数据库sql语言。mySql中用的是一张表Oracle中用的是序列所以这里要修改,不然会报错的噢。
<!--Oracle的配置方式 -->
<property name="dialect">org.hibernate.dialect.OracleDialect</property>
<!--mySql的配置方式-->
<property name="dialect">org.hibernate.dialect.MySQLDialect</property>
在Student.hbm.xml
中配置:
<id name="sid" type="java.lang.Integer" column="sid">
<!-- sequence走内部序列表,默认使用序列表hibernate_sequence -->
<generator class="sequence" />
</id>
指定序列表的话:
<generator class="sequence" >
<!-- 指定 aaa 这张表为序列表 -->
<param name="sequence_name">aaa</param>
</generator>
测试代码
DemoDao dao = new DemoDao();
Student stu = new Student();
stu.setSname("旺旺");
System.out.println(dao.addStudent(stu));
数据库中的sid为序列表生成
问题:
如果不指定生成的序列表那么就采用默认的那个表,所有会产生一个问题就是你可能多长表用一张序列表来生成主键可能造成断号
问题。
Hibernate控制
increment:最大id+1
在Student.hbm.xml
中配置:
<id name="sid" type="java.lang.Integer" column="sid">
<!-- 将最大id查出 + 1 数字,无需赋值 -->
<generator class="increment" />
</id>
测试代码
DemoDao dao = new DemoDao();
Student stu = new Student();
stu.setSname("旺旺");
System.out.println(dao.addStudent(stu));
Hibernate内部会做如下操作,将数据库中这长表的最大id查找出来 + 1来生成这次添加的主键.注意列段名最好为int类型
uuid/uuid.hex:生成随机编码
注意:由于随机编码是String类型的所有这里就在 Worker.hbm.xml配置
<id name="sid" type="java.lang.Integer" column="sid">
<!-- 随机数生成 32位的编码 -->
<generator class="uuid" />
</id>
特别说明:
uuid 随机数生成 32位的编码
uuid.hex 随机数生成 16位的编码
测试代码
DemoDao dao = new DemoDao();
Worker worker = new Worker();
worker.setWname("李旺旺");
System.out.println(dao.addWorker(worker));
主键就是随机数编码啦,例如像这样的4028ab816bee72a0016bee72a35f0000
一长串编码
自定义主键生成器
如果你对上面的主键生成都不太感兴趣的换,那么没关系我们还可以自定义主键生成
写自定义主键生成类(实现org.hibernate.id.IdentifierGenerator接口)
package com.cpc.tow.id;
import java.io.Serializable;
import java.text.SimpleDateFormat;
import java.util.Date;
import org.hibernate.HibernateException;
import org.hibernate.engine.spi.SharedSessionContractImplementor;
import org.hibernate.id.IdentifierGenerator;
public class Myts implements IdentifierGenerator{
@Override
public Serializable generate(SharedSessionContractImplementor session, Object object) throws HibernateException {
// TODO Auto-generated method stub
SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
//这里返回主键生成结果就ok啦
return "cpc:" + simpleDateFormat.format(new Date());
}
}
Worker.hbm.xml配置
<id name="sid" type="java.lang.Integer" column="sid">
<!-- 使用自定义的主键生成策略 -->
<generator class="com.cpc.tow.id.Myts" />
</id>
测试代码
DemoDao dao = new DemoDao();
Worker worker = new Worker();
worker.setWname("李旺旺");
System.out.println(dao.addWorker(worker));
生成的主键就是我们自己定义的啦