Hibernate 之 主键生成策略

本文介绍了Hibernate的主键生成策略,包括assigned、identity、sequence等六种常见类型,说明了各类型特点及适用数据库。还给出了Demo案例,涵盖util、entity、dao等部分,最后介绍了自定义主键生成器的方法,如在*.hbm.xml指定生成器类。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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:
在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值