做个网站需要什么/今天的新闻头条最新消息
主键生成策略
- 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&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 {@Beforepublic void setUp() throws Exception {System.out.println("测试所有要测试的方法之前都会执行的内容(一般用来初始化数据所用)");}@Afterpublic void tearDown() throws Exception {System.out.println("测试所有要测试的方法之后都会执行的内容(一般用来释放资源所用)");}@Testpublic void testAddStu() {System.out.println("要测试的方法一");}@Testpublic 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();@Beforepublic void setUp() throws Exception {
// System.out.println("测试所有要测试的方法之前都会执行的内容(一般用来初始化数据所用)");}@Afterpublic void tearDown() throws Exception {
// System.out.println("测试所有要测试的方法之后都会执行的内容(一般用来释放资源所用)");}@Testpublic void testAddStu() {Student stu = new Student();stu.setSname("小熊");demoDao.addStu(stu);// System.out.println("要测试的方法一");}@Testpublic void testAddWorker() {System.out.println("要测试的方法二");}}
Studen表的sid不是自增长
把Student.hbm.xml改
<generator class="increment" />
运行结果:
增加成功:
数据库表没有设置自增长还是能按顺序增加是因为
程序员控制
<generator class="assigned" />
对DemoDaoTest方法进行修改
@Testpublic void testAddStu() {Student stu = new Student();stu.setSname("小猪");
// 程序员控制,需要手动指定idstu.setSid(80);demoDao.addStu(stu);// System.out.println("要测试的方法一");}
选中运行打印:
增加成功:
当数据库表设置自增长
修改DemoDaoTest的方法:
@Testpublic void testAddStu() {Student stu = new Student();stu.setSname("小兔");
// 程序员控制,需要手动指定idstu.setSid(83);demoDao.addStu(stu);// System.out.println("要测试的方法一");}
提问:数据由上结果增长的是81还是83
答:83
右击运行结果:
增加成功是83:
因为设置了 由你自己控制和数据库表没有任何关系
Student.hbm.xml改成数据库控制
<generator class="identity" />
修改DemoDaoTest方法
@Testpublic void testAddStu() {Student stu = new Student();stu.setSname("小猫咪");
// 程序员控制,需要手动指定idstu.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的增加方法
@Testpublic void testAddWorker() {Worker worker = new Worker();worker.setWname("樱桃小丸子");worker.setWid("abcdef");demoDao.addWorker(worker);}
右击运行方法增加后表的数据:
Worker.hbm.xml改成uuid
<generator class="uuid" />
DemoDaoTest的Worker的方法
@Testpublic 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.TABLEPersistentIdentifierGenerator.PKassigned、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{@Overridepublic 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" />
运行结果:
增加成功: