Hibernate系列之ID生成策略
一、概述
hibernate中使用两种方式实现主键生成策略,分别是XML生成id和注解方式(@GeneratedValue),下面逐一进行总结。
二、XML配置方法
这种方式是在XX.hbm.xml文件中对generator进行配置,eg:
- <?xml version="1.0"?>
- <!DOCTYPE hibernate-mapping PUBLIC
- "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
- "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
- <hibernate-mapping package="com.test.demo">
- <class name="Student">
- <id name="id">
- <generator class="native"></generator>
- </id>
- <property name="name"></property>
- <property name="age"></property>
- </class>
- </hibernate-mapping>
常用的生成策略有以下几种:
identity:对DB2,Mysql,MS SQL Server等的内置标识字段提供支持,返回的标识符是long,short或者int类型
native:可以是identity类型、sequence类型或者hilo类型,取决于不同的底层数据库
sequence:在Oracle,SAP DB中使用序列(sequence)
uuid:使用一种128位的UUID算法产生的字符类型标识,像IP地址一样全网唯一
三、注解方式生成ID:@GeneratorValue
标准的annotation方式的主键生成策略如下:
- AUTO:可以是identity类型或者是sequence类型或者是table类型,取决于底层的数据库
- import javax.persistence.Entity;
- import javax.persistence.GeneratedValue;
- import javax.persistence.GenerationType;
- import javax.persistence.Id;
- @Entity
- public class Person
- {
- private String name;
- private int age;
- private int id;
- public String getName()
- {
- return name;
- }
- public void setName(String name)
- {
- this.name = name;
- }
- public int getAge()
- {
- return age;
- }
- public void setAge(int age)
- {
- this.age = age;
- }
- @Id
- @GeneratedValue(strategy=GenerationType.AUTO)
- public int getId()
- {
- return id;
- }
- public void setId(int id)
- {
- this.id = id;
- }
- }
- TABLE:使用表保存id值,即会为应用的表创建一张专门保存id的表
- import javax.persistence.Entity;
- import javax.persistence.GeneratedValue;
- import javax.persistence.GenerationType;
- import javax.persistence.Id;
- import javax.persistence.TableGenerator;
- @Entity
- public class Person
- {
- private String name;
- private int age;
- private int id;
- public String getName()
- {
- return name;
- }
- public void setName(String name)
- {
- this.name = name;
- }
- public int getAge()
- {
- return age;
- }
- public void setAge(int age)
- {
- this.age = age;
- }
- @Id
- @TableGenerator(name="personID",table="personID_DB",pkColumnName="key_value",pkColumnValue="pk_value",valueColumnName="person",allocationSize=1)
- @GeneratedValue(strategy=GenerationType.TABLE,generator="personID")
- public int getId()
- {
- return id;
- }
- public void setId(int id)
- {
- this.id = id;
- }
- }
- IDENTITY:identity column
- import javax.persistence.Entity;
- import javax.persistence.GeneratedValue;
- import javax.persistence.GenerationType;
- import javax.persistence.Id;
- import javax.persistence.TableGenerator;
- @Entity
- public class Person
- {
- private String name;
- private int age;
- private int id;
- public String getName()
- {
- return name;
- }
- public void setName(String name)
- {
- this.name = name;
- }
- public int getAge()
- {
- return age;
- }
- public void setAge(int age)
- {
- this.age = age;
- }
- @Id
- @GeneratedValue(strategy=GenerationType.IDENTITY)
- public int getId()
- {
- return id;
- }
- public void setId(int id)
- {
- this.id = id;
- }
- }
- SEQUENCE:sequence
四、联合主键生成策略
有的时候我们需要将一个实体的2个或多个字段联合起来作为主键,就是说,不能有2个或多个对象的这几个字段值都相同的情况发生。现在我们要将Person字段的id和name字段联合作为主键:
- @Entity
- public class Person
- {
- //现在id和name组成联合主键
- private int id;
- private String name;
- private int age;
- ...
- }
- 首先将联合主键的属性提取出来,重新编写一个pojo类(原pojo类中的id,name要删除 并新加入属性“PersonPK”)
- 新建pojo类必须实现 java.io.Serializable 序列化接口
- 新pojo类要重写equals和hashCode方法
- public class PersonPK implements Serializable
- {
- private String name;
- private int id;
- public String getName()
- {
- return name;
- }
- public void setName(String name)
- {
- this.name = name;
- }
- public int getId()
- {
- return id;
- }
- public void setId(int id)
- {
- this.id = id;
- }
- @Override
- public int hashCode()
- {
- return this.name.hashCode();
- }
- @Override
- public boolean equals(Object obj)
- {
- if(obj instanceof PersonPK) {
- PersonPK pk = (PersonPK)obj;
- if(this.id == pk.getId() && this.name.equals(pk.getName())) {
- return true;
- }
- }
- return false;
- }
- }
联合主键生成策略XML配置方法:
- <?xml version="1.0"?>
- <!DOCTYPE hibernate-mapping PUBLIC
- "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
- "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
- <hibernate-mapping package="com.test.demo">
- <class name="Person">
- <composite-id name="personPK" class="com.test.demo.PersonPK">
- <key-property name="id"></key-property>
- <key-property name="name"></key-property>
- </composite-id>
- <property name="age" />
- </class>
- </hibernate-mapping>
联合主键ID生成策略的Annotation版本,共有三种方式,前三步骤一样,另外:
方法1、在新类PersonPK前写@Embeddable,在原Person类的新属性PersonPK的get方法前写@id
- @Embeddable
- public class PersonPK implements Serializable
- {
- private static final long serialVersionUID = -7068850328521576106L;
- private String name;
- private int id;
- public String getName()
- {
- return name;
- }
- public void setName(String name)
- {
- this.name = name;
- }
- public int getId()
- {
- return id;
- }
- public void setId(int id)
- {
- this.id = id;
- }
- @Override
- public int hashCode()
- {
- return this.name.hashCode();
- }
- @Override
- public boolean equals(Object obj)
- {
- if(obj instanceof PersonPK) {
- PersonPK pk = (PersonPK)obj;
- if(this.id == pk.getId() && this.name.equals(pk.getName())) {
- return true;
- }
- }
- return false;
- }
- }
Person类中:
- @Entity
- public class Person
- {
- private PersonPK personPK;
- private int age;
- public int getAge()
- {
- return age;
- }
- public void setAge(int age)
- {
- this.age = age;
- }
- @Id
- public PersonPK getPersonPK()
- {
- return personPK;
- }
- public void setPersonPK(PersonPK personPK)
- {
- this.personPK = personPK;
- }
- }
方法2、新类无需添加注解,只需在原类Person新属性PersonPK的get方法前写@EmbeddID即可
- @Entity
- public class Person
- {
- private PersonPK personPK;
- private int age;
- public int getAge()
- {
- return age;
- }
- public void setAge(int age)
- {
- this.age = age;
- }
- @EmbeddedId
- public PersonPK getPersonPK()
- {
- return personPK;
- }
- public void setPersonPK(PersonPK personPK)
- {
- this.personPK = personPK;
- }
- }
方法3、新pojo类无需加注解,原pojo类的id,name属性保留不变,也无需新增“TercherPK”属性。 只在id,name的get方法前都加@Id,并在原pojo类前加@IdClass(PersonPK.class):
原类Person:
- @Entity
- @IdClass(PersonPK.class)
- public class Person
- {
- private int age;
- private String name;
- private int id;
- @Id
- public String getName()
- {
- return name;
- }
- public void setName(String name)
- {
- this.name = name;
- }
- @Id
- public int getId()
- {
- return id;
- }
- public void setId(int id)
- {
- this.id = id;
- }
- public int getAge()
- {
- return age;
- }
- public void setAge(int age)
- {
- this.age = age;
- }
- }
运行测试程序(针对上述三种方法,测试用例需要稍作修改,这里不在赘述):
- public class PersonTest
- {
- private static SessionFactory sf=null;
- @BeforeClass
- public static void beforeClass()
- {
- sf=new AnnotationConfiguration().configure().buildSessionFactory();
- }
- @Test
- public void test()
- {
- PersonPK personPK=new PersonPK();
- personPK.setId(1);
- personPK.setName("xujian");
- Person p=new Person();
- p.setAge(23);
- p.setPersonPK(personPK);
- Session session=sf.openSession();
- session.beginTransaction();
- session.save(p);
- //提交事物
- session.getTransaction().commit();
- session.close();
- sf.close();
- }
- @AfterClass
- public static void afterClass()
- {
- sf.close();
- }
- }
可以看到:
生成的Person表中id和name组成联合主键
Hibernate系列之ID生成策略的更多相关文章
- 图解Janusgraph系列-分布式id生成策略分析
JanusGraph - 分布式id的生成策略 大家好,我是洋仔,JanusGraph图解系列文章,实时更新~ 本次更新时间:2020-9-1 文章为作者跟踪源码和查看官方文档整理,如有任何问题,请联 ...
- Hibernate 再接触 ID生成策略
Xml 方法 在student.hbm.xml中 <generator class="uuid"></generator> 取值如下 1.identity: ...
- hibernate(四)ID生成策略
一.ID生成策略配置 1.ID生成方式在xml中配置方式: <?xml version="1.0"?> <!DOCTYPE hibernate-mapping P ...
- hibernate 中id生成策略
数据库的设计和操作中,我们通常会给表建立主键. 主键,可以分为自然主键和代理主键. 自然主键表示:采用具有业务逻辑含义的字段作为表的主键.比如在用户信息表中,采用用户的身份证号码作为主键.但是这样一来 ...
- Rhythmk 学习 Hibernate 03 - Hibernate 之 延时加载 以及 ID 生成策略
Hibernate 加载数据 有get,跟Load 1.懒加载: 使用session.load(type,id)获取对象,并不读取数据库,只有在使用返回对象值才正真去查询数据库. @Test publ ...
- [Hibernate开发之路](4)ID生成策略
一 对象关系数据库映射之Id 被映射的类必须定义相应数据库表主键字段.大多数类有一个JavaBeans风格的属性, 为每个实例包括唯一的标识. <id> 元素定义了该属性到数据库表主键字段 ...
- hibernate ID生成策略配置
1.Student.hbm.xml配置 <hibernate-mapping package="com.wxh.hibernate.model"> <class ...
- hibernate注解主键生成策略
Id生成策略: @GeneratedValue,JPA通用策略生成器 . JPA提供的四种标准用法为TABLE,SEQUENCE,IDENTITY,AUTO. TABLE:使用一个特定的数据库表格来 ...
- JPA ID生成策略(转---)
尊重原创:http://tendyming.iteye.com/blog/2024985 JPA ID生成策略 @Table Table用来定义entity主表的name,catalog,schema ...
随机推荐
- SAP 以工序为基准进行发料 机加工行业 Goods Issue to Routing
SAP 以工序为基准进行发料 这个流程是在业务有关需求,业务需要按照工序发料,一个工单有多个工序,而料是要发到每个工序上,而且没到工序之间在物理上是有距离的,所以仓管员在打印配发单之后希望了解到哪 ...
- e777. 获得JList组件的所有项
// Create a list String[] items = {"A", "B", "C", "D"}; JLis ...
- CI框架 -- 密码哈希
哈希算法是一个单向函数.它可以将任何大小的数据转化为定长的“指纹”,并且无法被反向计算 依赖性 crypt() 函数需支持 CRYPT_BLOWFISH 常量 PASSWORD_BCRYPT PASS ...
- C# 获取web.config配置文件
.ConfigurationManager提供对客户端应用程序配置文件的访问. 其有两个属性1.ConnectionStrings 获取当前应用程序默认配置的 ConnectionStringsSec ...
- Drools 语法
Drools 语法 规则语法 package: package 的名字是随意的,不必必须对应物理路径 import: 导入外部变量 规则的编译与运行要通过Drools 提供的各种API 来实现.API ...
- Material Design Support 8大控件介绍
TextInputLayout 显示提示信息 能够通过调用setError()在EditText以下显示一条错误信息 FloatingActionButton 悬浮操作按钮 Snackbar 相当于底 ...
- php public,static,private,protected,final,const,abstract
public:权限是最大的,可以内部调用,实例调用等. protected: 受保护类型,用于本类和继承类调用. private: 私有类型,只有在本类中使用. final:PHP 5:不被改,不被继 ...
- [转]jmeter 自定义测试脚本
http://blog.csdn.net/kash_chen007/article/details/37690411 http://wangym.iteye.com/blog/731729 1.创建一 ...
- 在window的cmd窗口下运行linux命令
之前看很多视频老师都是用Linux命令操作命令框,感觉很方便,自己在cmd窗口试了一下,所有这些命令都提示不是内部或外部命令,后来发现了windows还有一个powershell命令行工具,用起来似乎 ...
- QT编译错误: multiple definition of `qMain(int, char**)'
QT使用过程中来回添加修改代码,结果出现了编译错误:error: multiple definition of `qMain(int, char**)' 一直看我的源文件是都哪里有错误,最后发现是在p ...