Hibernate主键生成策略
六种常见的主键类型
assigned:
assigned主键由外部程序生成,在save()之前必须赋值。assigned主键类型与HIbernate和底层数据库都无关,所以可以跨数据库操作。
identity:
identity主键类型由底层数据库自己生成,但是必须设为自动增长类型。使用identity主键的前提条件就是数据库支持自动增长字段类型,例如SQLServer,MySql,DB2,Sybase等。Oracle不支持自动增长字段类型,则不能使用identity主键类型。要设定sequence
sequence:
sequence主键类型是采用数据库提供的sequence机制生成的主键,需要数据库支持sequence类型,例如Oracle,DB,SAP DB,PostgerSQL,Mckoi等,而MySql不支持sequence类型。但没有指定具体的sequence的时候,Hibernate会访问一个默认的sequence,是hibernate_sequence,我们也需要在数据库中建立这个sequence
<!-- 解决hibernate在Mysql中无法自动建表的问题 -->
<property name="dialect">org.hibernate.dialect.MySQL5InnoDBDialect</property>
<property name="hbm2ddl.auto">update</property>
increment:
increment主键类型由Hibernate向数据库获取最大值(每个session获取以一次),以这个最大值为基础,每次增量加1,生成主键,不依赖于数据库,可以进行跨数据库操作。
uuid:
使用一个128-bit的UUID算法生成字符串类型的标识符,UUID被编码成一个32位16进制数字的字符串。UUID包含:IP地址、JVM启动时间、系统时间(精确到1/4秒)和一个计数器值(JVM中唯一)
hibernate会算出一个128位的唯一值插入
uuid.string
hibernate会算出一个16位的值插入
唯一缺点就是生成的结果串会比较长
native:
等于identity+sequence
native主键类型由Hibernate根据使用数据库自行判断采用identity,sequence,Hilo其中一种作为主键生成方式,灵活性很强。
Demo案例
util
SessionFactoryUtils
package com.li.two.util;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;
/**
* 这个类是在学习hibernate 的过程中所用 (整合SSH框架之前)
*
* 作用:
* 可以用来检测你所写的映射文件是否正确 90%
*
* @author Dragon
*
*/
public class SessionFactoryUtils {
private static SessionFactory sessionFactory;
// 存放当前会话
private static ThreadLocal<Session> threadLocal = new ThreadLocal<Session>();
static {
Configuration cfg = new Configuration();
Configuration configure = cfg.configure("/hibernate.cfg.xml");
sessionFactory = configure.buildSessionFactory();
}
public static Session openSession() {
Session session = threadLocal.get();
if (null == session) {
session = sessionFactory.openSession();
threadLocal.set(session);
}
return session;
}
public static void closeSession() {
Session session = threadLocal.get();
if (null != session) {
if (session.isOpen()) {
session.close();
}
threadLocal.set(null);
}
}
public static void main(String[] args) {
Session session = openSession();
System.out.println(session.isConnected());
closeSession();
}
}
entity
实体类属性对应数据库
Student
package com.li.two.entity;
import java.io.Serializable;
public class Student implements Serializable{
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 + "]";
}
}
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.li.two.entity.Student" table="t_hibernate_student">
<id name="sid" type="java.lang.Integer" column="sid">
<!-- <generator class="assigned" /> -->
<generator class="increment" />
<!-- <generator class="identity" /> -->
<!-- <generator class="sequence" > <param name="sequence_name">aaa</param>
</generator> -->
<!-- <generator class="com.javaxl.two.id.Myts" /> -->
</id>
<property name="sname" type="java.lang.String" column="sname">
</property>
</class>
</hibernate-mapping>
Worker
package com.li.two.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 + "]";
}
}
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.li.two.entity.Worker" table="t_hibernate_worker">
<id name="wid" type="java.lang.String" column="wid">
<!-- <generator class="assigned" /> -->
<!-- <generator class="uuid" /> -->
<!-- <generator class="sequence" > <param name="sequence_name">aaa</param>
</generator> -->
<generator class="com.li.two.id.Myts" />
</id>
<property name="wname" type="java.lang.String" column="wname">
</property>
</class>
</hibernate-mapping>
在hibernate.cfg.xml中配置
<!-- 主键生成策略 -->
<mapping resource="com/li/two/entity/Student.hbm.xml"/>
<mapping resource="com/li/two/entity/Worker.hbm.xml"/>
dao
DemoDao
package com.li.two.dao;
import java.io.Serializable;
import org.hibernate.Session;
import org.hibernate.Transaction;
import com.li.two.entity.Student;
import com.li.two.entity.Worker;
import com.li.two.util.SessionFactoryUtils;
public class DemoDao {
/**
* 添加学生
* @param student
* @return
*/
public Serializable addStu(Student student){
Session session = SessionFactoryUtils.openSession();
Transaction transaction = session.beginTransaction();
Serializable save = session.save(student);
transaction.commit();
session.close();
return save;
}
/**
* 添加工人
* @param worker
* @return
*/
public Serializable addWork(Worker worker){
Session session = SessionFactoryUtils.openSession();
Transaction transaction = session.beginTransaction();
Serializable save = session.save(worker);
transaction.commit();
session.close();
return save;
}
public static void testStudent(String[] args) {
DemoDao dao = new DemoDao();
Student stu = new Student();
// stu.setSid(1);
stu.setSname("XXX");
System.out.println(dao.addStu(stu));
}
public static void main(String[] args) {
DemoDao dao = new DemoDao();
Worker worker = new Worker();
worker.setWname("XXX");
System.out.println(dao.addWork(worker));
}
}
自定义主键生成器
再 *.hbm.xml指定主键生成器类
<generator class="xxx.MyTsGenerator"/>
在上面Worker.hbm.xml中已经配置了Myts类
package com.li.two.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 {
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
return "li_shop_book_"+sdf.format(new Date());
}
}
运行DemoDao: