1. hibernate的主键生成器:
generator元素:表示了一个主键生成器,它用来为持久化类实例生成唯一的标识 。
1.1 程序员自己控制:assigned
1.2 数据库控制: identity(标识列/自动增长) sequence
1.3 hibernate控制:increment uuid/uuid.hex
1.4 其它:native
公司名_模块名_时间戳
student id int
worker id varchar
2. 主键生成器要求
2.1 assigned
数据类型不限、保存前必须赋值
2.2 identity(重点掌握)
数字,无需赋值
2.3 sequence(重点掌握)
数字,无需赋值, 默认使hibernate_sequence这个序列,
也可以通过sequence/sequence_name参数赋值
2.4 increment
数字,无需赋值
2.5 uuid/uuid.hex (是由容器自动生成的一个32位的字符串,.hex代表的是十六进制)
32位的字符串,无需赋值,
2.6 native(重点掌握)
等于identity+sequence
- 自定义主键生成器
3.1 *.hbm.xml指定主键生成器类
以下是 主键生成器的案例:
创建工具包相应的实体类:
SessionFactoryUtils工具类:
package com.xiaoyi.two.util;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;
/**
*本工具类的作用:
*1.简化代码的书写
*2.检测hibernate相关配置是否正确
*/
public class SessionFactoryUtils {
private static SessionFactory sessionFactory;
static {
Configuration cfg = new Configuration().configure("/hibernate.cfg.xml");
sessionFactory = cfg.buildSessionFactory();
}
public static Session openSession() {
Session session = sessionFactory.getCurrentSession();
if(session == null) {
session = sessionFactory.openSession();
}
return session;
}
public static void closeSession() {
Session session = sessionFactory.getCurrentSession();
if(session != null && session.isOpen()) {
session.close();
}
}
public static void main(String[] args) {
Session session = SessionFactoryUtils.openSession();
session.beginTransaction();
System.out.println(session.isConnected());
SessionFactoryUtils.closeSession();
System.out.println(session.isConnected());
}
}
Student实体类:
package com.xiaoyi.two.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实体类:
package com.xiaoyi.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 + "]";
}
}
Student学生实体类的配置文件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.xiaoyi.two.entity.Student" table="t_hibernate_student">
<id name="sid" type="java.lang.Integer" column="sid">
<!-- hibernate一共四种生成 -->
<!-- <generator class="identity" /> --><!-- 3.由数据库表控制-->
<!-- <generator class="assigned"/> --><!--1.有程序员控制 id是否插入数据库表 -->
<!-- <generator class="increment" /> --> <!--2.由hibernate控制 -->
<generator class="sequence" /> <!-- 3.由数据库表控制-->
<!-- <generator class="sequence" > <param name="sequence_name">aaa</param>
</generator> -->
<!-- <generator class="com.xiaoyi.two.id.Myts" /> -->
</id>
<property name="sname" type="java.lang.String" column="sname">
</property>
</class>
</hibernate-mapping>
Worker工人实体类的配置文件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.xiaoyi.two.entity.Worker" table="t_hibernate_worker">
<id name="wid" type="java.lang.String" column="wid">
<!-- <generator class="uuid" /> -->
<!-- <generator class="assigned" /> --><!--自己控制 -->
<!-- <generator class="sequence" /> -->
<!-- <generator class="sequence" > <param name="sequence_name">aaa</param>
</generator> -->
<generator class="com.xiaoyi.two.id.Myts" /><!--自定义主键生成器 -->
</id>
<property name="wname" type="java.lang.String" column="wname">
</property>
</class>
</hibernate-mapping>
DemoDao:
package com.xiaoyi.two.dao;
import java.io.Serializable;
import org.hibernate.Session;
import org.hibernate.Transaction;
import com.xiaoyi.two.entity.Student;
import com.xiaoyi.two.entity.Worker;
import com.xiaoyi.two.util.SessionFactoryUtils;
/**添加学生
* @author Administrator
*
*/
public class DemoDao {
public Serializable addStu(Student stu) {
//SessionFactoryUtils中已经简化了代码,只需要获取session
Session session = SessionFactoryUtils.openSession();
//打开事务
Transaction transaction = session.beginTransaction();
// 操作数据库 Serializable 返回的是id
Serializable save = session.save(stu);
//提交事务
transaction.commit();
//关闭
SessionFactoryUtils.closeSession();
return save;
}
/**
* 添加工人
* @param worker
* @return
*/
public Serializable addWorker(Worker worker) {
Session session = SessionFactoryUtils.openSession();
Transaction transaction = session.beginTransaction();
Serializable save=session.save(worker);
transaction.commit();
SessionFactoryUtils.closeSession();
return save;
}
}
以下为测试类 DemoDaoTest:
package com.xiaoyi.two.dao;
import static org.junit.Assert.*;
import java.io.Serializable;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import com.xiaoyi.two.entity.Student;
import com.xiaoyi.two.entity.Worker;
public class DemoDaoTest {
private DemoDao demoDao = new DemoDao();
@Before
public void setUp() throws Exception {
//System.out.println("测试所有要测试的方法之前都会执行的内容(一般用来初始化数据所用)");
}
@After
public void tearDown() throws Exception {
// System.out.println("测试所有要测试的方法之后都会执行的内容(一般用来释放资源所用)");
}
@Test
public void testAddStu() {
Student stu=new Student();
stu.setSname("琪琪5");
stu.setSid(88);//由程序员控制要自己写id
demoDao.addStu(stu);
}
@Test
public void testAddWorker() {
Worker worker=new Worker();
worker.setWname("呵呵哒1");
worker.setWid("abcdefgh");//由自己控制要写id
//新增一个id
Serializable wid= demoDao.addWorker(worker);
demoDao.addWorker(worker);
System.out.println(wid);
}
}
测试所有结果:
Student.hbm.xml由sequence 数据库控制:
<generator class="sequence" /> <!-- 3.由数据库表控制-->
自己控制还要在里面设置id:
worker.hbm.xml里为自己控制(自己控制需要在测试中设置id):
uuid随机生产串:
<generator class="uuid" />
由不同数据库方言生产不同的sql语句:
(oraclec 里面临时表的序列号,在主配置文件里面加oracle路径)
创建主键生成器类:
实现org.hibernate.id.IdentifierGenerator接口即可,并还可以实现org.hibernate.id.Configurable接口来读取一些配置信息
PersistentIdentifierGenerator.TABLE
PersistentIdentifierGenerator.PK
assigned、native、自定义主键
案例 Myts:
package com.xiaoyi.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 {
//Serializable返回id
@Override
public Serializable generate(SharedSessionContractImplementor session, Object object) throws HibernateException {
//日期格式
SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
return "book_category_"+sdf.format(new Date());
}
}
在worker.hbm.xml中配置自定义主键生成器 :
<generator class="com.xiaoyi.two.id.Myts" /><!--自定义主键生成器 -->
结果: