hibernate之主键生成策略

首先我们学习hibernate需要用到一个工具类

package com.myy.two.util;

import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;

/**
 * 本工具类的作用:
 * 1.简化代码
 * 2.检测hibernate相关的配置是否正确
 * @author myy
 *
 */
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());
	}
}

代码块读取主配置文件,获取工厂类。对外暴露了两个方法,开启和关闭session。意味着如果想获取session对象只需要通过这个工具包调用方法。

Student

package com.myy.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.myy.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.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.myy.two.entity.Student" table="t_hibernate_student">
		<id name="sid" type="java.lang.Integer" column="sid">
			<generator class="assigned" />
<!-- 			<generator class="increment" /> -->
			<!-- <generator class="sequence" /> -->
			<!-- <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.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.myy.two.entity.Worker" table="t_hibernate_worker">
		<id name="wid" type="java.lang.String" column="wid">
			<generator class="assigned" />
			<!-- <generator class="sequence" /> -->
			<!-- <generator class="sequence" > <param name="sequence_name">aaa</param> 
				</generator> -->
			<!-- <generator class="com.javaxl.two.id.Myts" /> -->
		</id>

		<property name="wname" type="java.lang.String" column="wname">
		</property>
	</class>
</hibernate-mapping>

映射文件的配置
hibernate.cfg.xml

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-configuration PUBLIC
	"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
	"http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
	<session-factory>
		<!-- 1. 数据库相关 -->
		<property name="connection.username">root</property>
		<property name="connection.password">123</property>
		<property name="connection.url">jdbc:mysql://localhost:3306/db_xm?useUnicode=true&amp;characterEncoding=UTF-8
		</property>
		<property name="connection.driver_class">com.mysql.jdbc.Driver</property>
		<property name="dialect">org.hibernate.dialect.MySQLDialect</property>

		<!-- 配置本地事务(No CurrentSessionContext configured!) -->
		<property name="hibernate.current_session_context_class">thread</property>

		<!-- 2. 调试相关 -->
		<property name="show_sql">true</property>
		<property name="format_sql">true</property>

		<!-- 3. 添加实体映射文件 -->
		<mapping resource="com/myy/one/entity/User.hbm.xml"/>
		<!-- 讲解主键生产策略 -->
		<mapping  resource="com/myy/two/entity/Student.hbm.xml"/>
		<mapping  resource="com/myy/two/entity/Worker.hbm.xml"/>
	</session-factory>
</hibernate-configuration>

SessionFactoryUtils工具类的检测hibernate相关的配置是否正确
运行代码:
在这里插入图片描述

写错了就会报错
在这里插入图片描述

jUnit针对于你要测试的类,每一个类都去生产一个测试方法

==问:==如何操作
答:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
DemoDao

package com.myy.two.dao;

import static org.junit.Assert.*;

import org.junit.After;
import org.junit.Before;
import org.junit.Test;

public class DemoDaoTest {

	@Before
	public void setUp() throws Exception {
		System.out.println("测试所有要测试的方法之前都会执行的内容(一般用来初始化数据所用)");
	}

	@After
	public void tearDown() throws Exception {
		System.out.println("测试所有要测试的方法之后都会执行的内容(一般用来释放资源所用)");

	}

	@Test
	public void testAddStu() {
		System.out.println("要测试的方法一");
	}

	@Test
	public void testAddWorker() {
		System.out.println("要测试的方法二");
	}

}

鼠标选中右击运行结果:
在这里插入图片描述
如果当没有选中测试方法时直接运行则会测试所有的测试方法:
在这里插入图片描述

如果单个测试方法报错,是因为你并没有选中你要测试的方法,你测试了所有的方法

hibernate的主键生成器:

generator元素:表示了一个主键生成器,它用来为持久化类实例生成唯一的标识 。
1 程序员自己控制:assigned
2 数据库控制: identity(标识列/自动增长) sequence
3 hibernate控制:increment uuid/uuid.hex
4 其它:native
  公司名_模块名_时间戳
package com.myy.two.dao;

import static org.junit.Assert.*;

import org.junit.*;

import com.myy.two.entity.Student;

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("小熊");
		demoDao.addStu(stu);
		
//		System.out.println("要测试的方法一");
	}

	@Test
	public void testAddWorker() {
		System.out.println("要测试的方法二");
	}

}

Studen表的sid不是自增长
把Student.hbm.xml改

<generator class="increment" />

在这里插入图片描述

运行结果:
在这里插入图片描述
增加成功:
在这里插入图片描述

数据库表没有设置自增长还是能按顺序增加是因为

程序员控制

<generator class="assigned" />

在这里插入图片描述
对DemoDaoTest方法进行修改

	@Test
	public void testAddStu() {
		Student stu = new Student();
		stu.setSname("小猪");
//		程序员控制,需要手动指定id
		stu.setSid(80);
		demoDao.addStu(stu);
		
//		System.out.println("要测试的方法一");
	}

选中运行打印:
在这里插入图片描述
增加成功:
在这里插入图片描述
当数据库表设置自增长
修改DemoDaoTest的方法:

	@Test
	public void testAddStu() {
		Student stu = new Student();
		stu.setSname("小兔");
//		程序员控制,需要手动指定id
		stu.setSid(83);
		demoDao.addStu(stu);
		
//		System.out.println("要测试的方法一");
	}

提问:数据由上结果增长的是81还是83
:83
右击运行结果:
在这里插入图片描述
增加成功是
83

在这里插入图片描述

因为设置了 由你自己控制和数据库表没有任何关系

Student.hbm.xml改成数据库控制

<generator class="identity" />

修改DemoDaoTest方法

	@Test
	public void testAddStu() {
		Student stu = new Student();
		stu.setSname("小猫咪");
//		程序员控制,需要手动指定id
		stu.setSid(90);
		demoDao.addStu(stu);
		
//		System.out.println("要测试的方法一");
	}

右击运行增加后的ID是由上面的数据增长的:
在这里插入图片描述

和写的方法没有关系,由Student.hbm.xml有关系

主键生成器的要求

 assigned
      数据类型不限、保存前必须赋值
      
 identity(重点掌握)
      数字,无需赋值

sequence(重点掌握)
      数字,无需赋值, 默认使hibernate_sequence这个序列,
      也可以通过sequence/sequence_name参数赋值
      
increment
      数字,无需赋值

 uuid/uuid.hex (是由容器自动生成的一个32位的字符串,.hex代表的是十六进制)
      32位的字符串,无需赋值

 native(重点掌握)
      等于identity+sequence

DemoDaoTest的Worker的增加方法

	@Test
	public void testAddWorker() {
		Worker worker = new Worker();
		worker.setWname("樱桃小丸子");
		worker.setWid("abcdef");
		demoDao.addWorker(worker);
	}

右击运行方法增加后表的数据:
在这里插入图片描述

Worker.hbm.xml改成uuid

<generator class="uuid" />

在这里插入图片描述
DemoDaoTest的Worker的方法

	@Test
	public void testAddWorker() {
		Worker worker = new Worker();
		worker.setWname("樱桃小丸子");
		worker.setWid("abcdef11111");
		demoDao.addWorker(worker);
	}

右击运行后去的表数据:
在这里插入图片描述

加入的wid没有变化跟随设置的而改变,而是一个随机的字符串。常用

问:为什么hibernate支持跨数据库
Student.hbm.xml改成由hibernate控制:

<generator class="sequence" />

在这里插入图片描述
可在这里切换数据库方言:
在这里插入图片描述

答:根据不同的数据库方言生成不同的sql语句。意味着,换数据源,代码并不需要改变,用传统的jdbc,方法需要修改代码。

==提问:==在把数据插入表的时候,需要知道这条数据通过查看ID知道是哪个项目哪个模块什么时间点产生的数据。

自定义主键生成器

*.hbm.xml指定主键生成器类
      <generator class="xxx.MyTsGenerator"/> 

xxx.hbm.xml里面写的东西:例如:uuid,sequence,assigned等实际是一个类。

创建主键生成器类
      实现org.hibernate.id.IdentifierGenerator接口即可,并还可以实现org.hibernate.id.Configurable接口来读取一些配置信息
      PersistentIdentifierGenerator.TABLE
      PersistentIdentifierGenerator.PK
	  
	  
	  assigned、native、自定义主键

意味着,自己也可以写一个这样的类

package com.myy.two.entity;

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 sFormat= new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
	
		return "t_book_category_"+sFormat.format(new Date());
	}

}

修改hbm.xml

<generator class="com.myy.two.entity.Myts" />

在这里插入图片描述

运行结果:
在这里插入图片描述
增加成功:
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值