1. Hibernate支持三种继承映射策略:
–使用 subclass进行映射:将域模型中的每一个实体对象映射到一个独立的表中,也就是说不用在关系数据模型中考虑域模型中的继承关系和多态。
–使用 joined-subclass进行映射:对于继承关系中的子类使用同一个表,这就需要在数据库表中增加额外的区分子类类型的字段。
–使用 union-subclass
进行映射:域模型中的每个类映射到一个表,通过关系数据模型中的外键来描述表之间的继承关系。这也就相当于按照域模型的结构来建立数据库中的表,并通过外键来建立表之间的继承关系。

2.  采用subclass元素的继承映射:

•采用 subclass的继承映射可以实现对于继承关系中父类和子类使用同一张表
•因为父类和子类的实例全部保存在同一个表中,因此需要在该表内增加一列,使用该列来区分每行记录到低是哪个类的实例----这个列被称为辨别者列(discriminator).
•在这种映射策略下,使用 subclass来映射子类,使用class
或 subclass的
discriminator-value属性指定辨别者列的值
•所有子类定义的字段都不能有非空约束。如果为那些字段添加非空约束,那么父类的实例在那些列其实并没有值,这将引起数据库完整性冲突,导致父类的实例无法保存到数据库中

代码示例:

Person类

  1. public class Person {
  2.  
  3. private int id;
  4. private String name;
  5. private int age;
  6. public int getId() {
  7. return id;
  8. }
  9. public void setId(int id) {
  10. this.id = id;
  11. }
  12. public String getName() {
  13. return name;
  14. }
  15. public void setName(String name) {
  16. this.name = name;
  17. }
  18. public int getAge() {
  19. return age;
  20. }
  21. public void setAge(int age) {
  22. this.age = age;
  23. }
  24.  
  25. }

继承了Person类的Student类:

  1. public class Student extends Person {
  2. private String school;
  3.  
  4. public String getSchool() {
  5. return school;
  6. }
  7.  
  8. public void setSchool(String school) {
  9. this.school = school;
  10. }
  11.  
  12. }

Person类对应的hbm配置文件

  1. <?xml version="1.0"?>
  2. <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
  3. "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
  4. <!-- Generated 2015-10-26 22:01:46 by Hibernate Tools 3.4.0.CR1 -->
  5. <hibernate-mapping package="com.elgin.hibernate.subclass">
  6. <class name="Person" table="PERSON" discriminator-value="person">
  7. <id name="id" type="int">
  8. <column name="ID" />
  9. <generator class="native" />
  10. </id>
  11. <!-- 配置辨别者列 -->
  12. <discriminator column="TYPE" type="java.lang.String">
  13. </discriminator>
  14.  
  15. <property name="name" type="java.lang.String">
  16. <column name="NAME" />
  17. </property>
  18. <property name="age" type="int">
  19. <column name="AGE" />
  20. </property>
  21. <!-- 映射student子类 -->
  22. <subclass name="Student" discriminator-value="student">
  23. <property name="school" column="SCHOOL" type="java.lang.String"></property>
  24. </subclass>
  25. </class>
  26. </hibernate-mapping>

单元测试类:

  1. public class HibernateTest {
  2.  
  3. //如此声明只为方便测试,生产环境不能这么用
  4. private SessionFactory sessionFactory;
  5. private Session session;
  6. private Transaction transcation;
  7.  
  8. @Before
  9. public void init(){
  10. Configuration cfg=new Configuration().configure();
  11. ServiceRegistry serviceRegistry=new ServiceRegistryBuilder().applySettings(cfg.getProperties()).buildServiceRegistry();
  12. sessionFactory=cfg.buildSessionFactory(serviceRegistry);
  13. session=sessionFactory.openSession();
  14. transcation=session.beginTransaction();
  15. }
  16. /*
  17. * subclass继承映射缺点
  18. * 1.加入了辨别者列
  19. * 2.子类独有的字段不能添加非空约束
  20. * 3.若继承层次较深,则数据表的字段也会较多
  21. *
  22. */
  23.  
  24. /**
  25. * 插入操作:
  26. * 1.对于子类,只需把数据插入到一张表中
  27. * 2.辨别者列由hibernate自动维护
  28. *
  29. */
  30. @Test
  31. public void testSave(){
  32. Person person=new Person();
  33. person.setName("peter");
  34. person.setAge(20);
  35.  
  36. Student student=new Student();
  37. student.setAge(22);
  38. student.setName("phne");
  39. student.setSchool("beida");
  40.  
  41. session.save(person);
  42. session.save(student);
  43. }
  44.  
  45. /**
  46. * 1.查询父类记录只需要查询一张数据表,结果包含子类数据
  47. * 2.查询子类记录同样之查询一张数据表,结果只有子类
  48. */
  49. @SuppressWarnings("unchecked")
  50. @Test
  51. public void testquery(){
  52. List<Person> persons=session.createQuery("from Person").list();
  53. System.out.println(persons.size());
  54.  
  55. List<Student> stus=session.createQuery("from Student").list();
  56. System.out.println(stus.size());
  57. }
  58.  
  59. @After
  60. public void destory(){
  61. transcation.commit();
  62. session.close();
  63. sessionFactory.close();
  64. }
  65.  
  66. }

3.  采用 joined-subclass元素的继承映射

•采用joined-subclass元素的继承映射可以实现每个子类一张表
•采用这种映射策略时,父类实例保存在父类表中,子类实例由父类表和子类表共同存储。因为子类实例也是一个特殊的父类实例,因此必然也包含了父类实例的属性。于是将子类和父类共有的属性保存在父类表中,子类增加的属性,则保存在子类表中。
•在这种映射策略下,无须使用鉴别者列,但需要为每个子类使用 key元素映射共有主键。
•子类增加的属性可以添加非空约束。因为子类的属性和父类的属性没有保存在同一个表中

代码示例:

Person类和Student类同上

Person类对应的hbm配置文件:

  1. <?xml version="1.0"?>
  2. <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
  3. "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
  4. <!-- Generated 2015-10-26 22:01:46 by Hibernate Tools 3.4.0.CR1 -->
  5. <hibernate-mapping package="com.elgin.hibernate.joined_subclass">
  6. <class name="Person" table="PERSON" discriminator-value="person">
  7. <id name="id" type="int">
  8. <column name="ID" />
  9. <generator class="native" />
  10. </id>
  11.  
  12. <property name="name" type="java.lang.String">
  13. <column name="NAME" />
  14. </property>
  15. <property name="age" type="int">
  16. <column name="AGE" />
  17. </property>
  18. <!-- 映射student子类 -->
  19. <joined-subclass name="Student" table="STUDENTS">
  20. <key column="PERSON_ID"></key>
  21. <property name="school" column="SCHOOL" type="java.lang.String"></property>
  22. </joined-subclass>
  23. </class>
  24. </hibernate-mapping>

单元测试类:

  1. public class HibernateTest {
  2.  
  3. //如此声明只为方便测试,生产环境不能这么用
  4. private SessionFactory sessionFactory;
  5. private Session session;
  6. private Transaction transcation;
  7.  
  8. @Before
  9. public void init(){
  10. Configuration cfg=new Configuration().configure();
  11. ServiceRegistry serviceRegistry=new ServiceRegistryBuilder().applySettings(cfg.getProperties()).buildServiceRegistry();
  12. sessionFactory=cfg.buildSessionFactory(serviceRegistry);
  13. session=sessionFactory.openSession();
  14. transcation=session.beginTransaction();
  15. }
  16. /*
  17. * joined-subclass继承映射优点:
  18. * 1.无需要加入辨别者列
  19. * 2.子类独有的字段能添加非空约束
  20. * 3.没有冗余的字段
  21. *
  22. */
  23.  
  24. /**
  25. * 插入操作:
  26. * 1.对于子类记录至少插入2张数据表
  27. *
  28. */
  29. @Test
  30. public void testSave(){
  31. Person person=new Person();
  32. person.setName("peter");
  33. person.setAge(20);
  34.  
  35. Student student=new Student();
  36. student.setAge(22);
  37. student.setName("phne");
  38. student.setSchool("beida");
  39.  
  40. session.save(person);
  41. session.save(student);
  42. }
  43.  
  44. /**
  45. * 1.查询父类记录,做左外连接查询2张表
  46. * 2.查询子类记录,做内连接查询2张表
  47. */
  48. @SuppressWarnings("unchecked")
  49. @Test
  50. public void testquery(){
  51. List<Person> persons=session.createQuery("from Person").list();
  52. System.out.println(persons.size());
  53.  
  54. List<Student> stus=session.createQuery("from Student").list();
  55. System.out.println(stus.size());
  56. }
  57.  
  58. @After
  59. public void destory(){
  60. transcation.commit();
  61. session.close();
  62. sessionFactory.close();
  63. }
  64.  
  65. }

4.  采用 union-subclass元素的继承映射

•采用 union-subclass元素可以实现将每一个实体对象映射到一个独立的表中。
•子类增加的属性可以有非空约束---即父类实例的数据保存在父表中,而子类实例的数据保存在子类表中。
•子类实例的数据仅保存在子类表中,而在父类表中没有任何记录
•在这种映射策略下,子类表的字段会比父类表的映射字段要多,因为子类表的字段等于父类表的字段、加子类增加属性的总和
•在这种映射策略下,既不需要使用鉴别者列,也无须使用key元素来映射共有主键.
•使用 union-subclass映射策略是不可使用identity的主键生成策略, 因为同一类继承层次中所有实体类都需要使用同一个主键种子,即多个持久化实体对应的记录的主键应该是连续的.受此影响,也不该使用native主键生成策略,因为native会根据数据库来选择使用identity或sequence.
  示例代码:
同样,Person类和Student类同上
Person类对应的hbm映射文件:
  1. <?xml version="1.0"?>
  2. <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
  3. "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
  4. <!-- Generated 2015-10-26 22:01:46 by Hibernate Tools 3.4.0.CR1 -->
  5. <hibernate-mapping package="com.elgin.hibernate.union_subclass">
  6. <class name="Person" table="PERSON" discriminator-value="person">
  7. <id name="id" type="int">
  8. <column name="ID" />
  9. <generator class="hilo" />
  10. </id>
  11.  
  12. <property name="name" type="java.lang.String">
  13. <column name="NAME" />
  14. </property>
  15. <property name="age" type="int">
  16. <column name="AGE" />
  17. </property>
  18. <!-- 映射student子类 -->
  19. <union-subclass name="Student" table="STUDENT">
  20. <property name="school" column="SCHOOL" type="java.lang.String"></property>
  21. </union-subclass>
  22. </class>
  23. </hibernate-mapping>

单元测试类:

  1. public class HibernateTest {
  2.  
  3. //如此声明只为方便测试,生产环境不能这么用
  4. private SessionFactory sessionFactory;
  5. private Session session;
  6. private Transaction transcation;
  7.  
  8. @Before
  9. public void init(){
  10. Configuration cfg=new Configuration().configure();
  11. ServiceRegistry serviceRegistry=new ServiceRegistryBuilder().applySettings(cfg.getProperties()).buildServiceRegistry();
  12. sessionFactory=cfg.buildSessionFactory(serviceRegistry);
  13. session=sessionFactory.openSession();
  14. transcation=session.beginTransaction();
  15. }
  16. /*
  17. * union-subclass继承映射优点:
  18. * 1.无需要加入辨别者列
  19. * 2.子类独有的字段能添加非空约束
  20. *
  21. * 缺点:
  22. * 1.存在冗余字段
  23. * 2.若更新父类表字段,更新效率低
  24. */
  25.  
  26. /**
  27. * 插入操作:
  28. *
  29. */
  30. @Test
  31. public void testSave(){
  32. Person person=new Person();
  33. person.setName("peter");
  34. person.setAge(20);
  35.  
  36. Student student=new Student();
  37. student.setAge(22);
  38. student.setName("phone");
  39. student.setSchool("北大");
  40.  
  41. session.save(person);
  42. session.save(student);
  43. }
  44.  
  45. /**
  46. * 1.查询父类记录,需要把父类记录和子类记录汇总到一起再做查询,性能稍差
  47. * 2.查询子类记录,只要查子类记录表即可
  48. */
  49. @SuppressWarnings("unchecked")
  50. @Test
  51. public void testquery(){
  52. List<Person> persons=session.createQuery("from Person").list();
  53. System.out.println(persons.size());
  54.  
  55. List<Student> stus=session.createQuery("from Student").list();
  56. System.out.println(stus.size());
  57. }
  58.  
  59. @Test
  60. public void testUpdate(){
  61. String hql="update Person p set p.age=45";
  62. session.createQuery(hql).executeUpdate();
  63. }
  64.  
  65. @After
  66. public void destory(){
  67. transcation.commit();
  68. session.close();
  69. sessionFactory.close();
  70. }
  71.  
  72. }

5.  3种继承映射方式的对比图






Hibernate之继承映射的更多相关文章

  1. hibernate笔记--继承映射关系的三种实现方式

    单表继承映射(一张表): 假设我们现在有三个类,关系如下: Person类有两个子类Student和Teacher,并且子类都具有自己独有的属性.这种实体关系在hibernate中可以使用单表的继承映 ...

  2. 初识Hibernate之继承映射

         前面的两篇文章中,我们介绍了两张表之间的各种相互关联映射关系,但往往我们也会遇到两张表甚至多张表之间共有着多个相同的字段.例如: 如图,student表和teacher表共同具有id,nam ...

  3. Hibernate的继承映射

    对象模型示例: 继承映射的实现方式有以下三种: (一)每棵类继承树一张表 (二)每个类一张表 (三)每个子类一张表 (一)每棵类继承树一张表 关系模型如下: 映射文件如下: <hibernate ...

  4. hibernate中继承映射保存

    1 简单继承映射,在子类上进行映射配置,可以将父类属性直接配置在子类映射文件中. 简单例子如下:teacher类继承自Person类. public class Person { private in ...

  5. hibernate映射-继承映射

    对于面向对象的程序设计语言而言,继承和多态是两个最基本的概念.Hibernate的继承映射可以理解成持久化类之间的继承关系.例如:人和学生之间的关系.学生继承人,可以认为学生是一个特殊的人,如果对人进 ...

  6. Hibernate4.x之映射关系--继承映射

    Hibernate的继承映射可以理解为持久化类之间的继承关系.例如:人和学生之间的关系.学生继承了人,可以认为学生是一个特殊的人,如果对人进行查询,学生的实例也将被得到. Hibernate支持以下三 ...

  7. 【Hibernate框架】三种继承映射

    一.综述 大家都知道,hibernate作为ORM框架的一个具体实现,最大的一个优点就是是我们的开发更加的能体现出"面向对象"的思想.在面向对象开发中,类与类之间是可以相互继承的( ...

  8. 【JavaEE】Hibernate继承映射,不用多态查询只查父表的方法

    几个月前,我在博问里面发了一个问题:http://q.cnblogs.com/q/64900/,但是一直没有找到好的答案,关闭问题以后才自己解决了,在这里分享一下. 首先我重复一下场景,博问里面举的动 ...

  9. Hibernate中的继承映射

    1.继承映射 继承映射分为两种情况:简单继承映射和继承映射. 在简单继承映射中,每个子类都要写一个映射文件. 在继承映射中,只使用一个映射文件.继承映射分为三种情况: 所有子类映射到一张表 需要使用鉴 ...

随机推荐

  1. HDU 3308 线段树 最长连续上升子序列 单点更新 区间查询

    题意: T个测试数据 n个数 q个查询 n个数 ( 下标从0开始) Q u v 查询 [u, v ] 区间最长连续上升子序列 U u v 把u位置改成v #include<iostream> ...

  2. c#模拟百度电击器方案

    核心提示: 我 们都知道百度对于用户体验很重视,如果一个关键词的某个搜索结果,点击量会很对的话,则百度会认为这个结果是用户所喜欢的结果,然后这个网站自然会在百度 得到一个很好的排名. 网络上也出现了百 ...

  3. decode-string(挺麻烦的)

    Java String作为参数传参是不会改变的,这个与常识的感觉不同. public String decodeString(String s) { s = ""; return ...

  4. bzoj4044

    这题简直了………… 首先根据操作可知,我们肯定是先造出某个偶数长度的回文串,然后添加若干字符得到设回文串长为len[x] 则ans=min(n-len[x]+f[x]); 那么问题就是制造这个串的回文 ...

  5. 【C#学习笔记】播放wma/mp3文件

    using System; using System.Runtime.InteropServices; namespace ConsoleApplication { class Program { [ ...

  6. zoj 3659 Conquer a New Region

    // 给你一颗树 选一个点,从这个点出发到其它所有点的权值和最大// i 到 j的最大权值为 i到j所经历的树边容量的最小值// 第一感觉是树上的dp// 后面发现不可以// 看了题解说是并查集// ...

  7. 【转】setTag()/getTag()

    原文网址:http://www.cnblogs.com/topcoderliu/archive/2011/06/07/2074419.html View中的setTag(Onbect)表示给View添 ...

  8. css3制作优惠券

    <div class="demo-container demo"><style> .demo{width:410px;} .stamp *{padding: ...

  9. ecshop 二次开发及模板标签

    ecs_account_log // 用户账目日志表   ecs_activity // 活动表(代码,名称,开始,结束,描述)   ecs_ad // 广告表(位置,类型,名称,链接,图片,开始,结 ...

  10. SQL注入中利用XP_cmdshell提权的用法(转)

    先来介绍一下子服务器的基本情况,windows 2000 adv server 中文版,据称 打过了sp3,asp+iis+mssql .首先扫描了一下子端口,呵呵,开始的一般步骤. 端口21开放: ...