`
neeleon
  • 浏览: 186137 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

hibernate入门(九)关联关系中的高级应用_继承关系

阅读更多

3. 继承关系

在前面的部门员工实例中,我们设定的员工只是普通的员工,现在假如有SaleSkill 两类员工,它们作为Employee 的子类。如何实现这些子类的映射?为了方便以后的操作说明和不影响以前的操作,我们把前面用到的员工部门类及相关实体配置文件,主配置文件等拷贝到一个新的项目theExtend 下进行操作。

1 )共享一张表:

意思是我们把子类的当作父类来处理共同映射成一张表。

>> 步骤一,创建Employee 的子类:SaleSkill 。内容如下:

package com.asm.hibernate.domain;

public class Sale extends Employee {

    private String signSale ;

    ... 省略 getXXX setXXX()

}

  package com.asm.hibernate.domain;

public class Skill extends Employee{

    private String signSkill ;

    ... 省略 getXXX setXXX()

}

>> 步骤二、修改 Employee.hbm.xml 配置文件:

< hibernate-mapping package = "com.asm.hibernate.domain" >

    < class name = "Employee" discriminator-value = "0" >

       < id name = "id" >

           < generator class = "native" />

       </ id >

       < discriminator column = "sign" type = "string" />

       < property name = "name" ></ property >

       < many-to-one name = "depart" column = "depart_id" />

        < subclass name = "Sale" discriminator-value = "1" >

           < property name = "signSale" ></ property >

       </ subclass >

       < subclass name = "Skill" discriminator-value = "2" >

           < property name = "signSkill" ></ property >

       </ subclass >

    </ class >

</ hibernate-mapping >

配置文件说明 理解 <subclass> 元素: 它作为class 的子元素,但是它却和class 非常相似,name 同样是指所关联到的“类”,下面的 < property > 也与class 下的 < property > 一样。我们注意到它和class 均增加了一个新属性 discriminator-value ,它的作用就是配置这些类的“鉴别类型”。而 discriminator-value 的配置主要参照的是 <discriminator> 它通常称为鉴别类型,即是说鉴别属于什么类型, 因为 在涉及到继承关系的操作时,总会涉及到父子类的关系,比如在上面的配置中,我们设定了sign 子段来标识类型,执行后会再对此作说明。 注意: < discriminator > 要写在后面元素的前面,因为它的dtd 文件是这样规定得。

>> 步骤三、修改主配置文件,由于是拷贝的前面的文件,所以需要去掉无关的映射文件,否则会提示找不到这此映射文件。

>> 步骤四、编写测试类:

package com.asm.hibernate.test;

public class ManyToOneTest {

    public static void main(String[] args) {

       add ();

       Employee emp =query (2);

       System. out .println( "emp type:" +emp );

    }

     static Employee query( int empId) {

       Session s = null ;

       try {

           s = HibernateUtil.getSession ();

           Employee emp = (Employee) s.get(Employee. class , empId);

           System. out .println( "Department Name:" + emp.getDepart().getName());

           return emp;

       } finally {

           if (s != null )

              s.close();

       }

    }

     static void add() {

       Session s = null ;

       Transaction tx = null ;

       try {

           Department depart = new Department();

           depart.setName( "departName" );

 

           Employee emp = new Employee();

           emp.setName( "empName" );

           emp.setDepart(depart);

            Sale emp2 = new Sale();

           emp2.setName( "saleEmployee" );

           emp2.setSignSale( "saleName" );

           emp2.setDepart(depart);

            Skill emp3 = new Skill();

           emp3.setName( "skillEmployee" );

           emp3.setSignSkill( "skillName" );

           emp3.setDepart(depart);

            s = HibernateUtil.getSession ();

           tx = s.beginTransaction();

            s.save(emp);

           s.save(emp2);

           s.save(emp3);

           s.save(depart);

           tx.commit();

       } finally {

           if (s != null )

              s.close();

       }

    }

}

说明:没的什么可多言的,只是要注意在查询时能返回其子类型。

下面来看执行后employee 表的内容:
+----+------+---------------+-----------+----------+-----------+

| id | sign | name          | depart_id | signSale | signSkill |

+----+------+---------------+-----------+----------+-----------+

|  1 | 0    | empName       |         1 | NULL     | NULL      |

|  2 | 1    | saleEmployee  |         1 | saleName | NULL      |

|  3 | 2    | skillEmployee |         1 | NULL     | skillName |

+----+------+---------------+-----------+----------+-----------+
先来看sign 这列:由于sign 是鉴别类型设定的字段,且分别在前面为EmployeeSaleSkill 分别配置了“012 ”所以它们会在sign 体现出来。其实 < discriminator column = "sign" type = "string" /> 也可以把type 属性值设为“int ”等。
再来看“signSalesignSkill ”字段:它们本身是专为特定的类的属性配置的字段(比如在第一个<subclass> 元素下的property 子元素就配置了Sale 类的signSale 属性):所以它们只适合特定的类,而不适的类将会以null 来填充,这也就是如果采取“共享一张表”的最大缺点,它限制了我们不能在子类属性所映射的字段上设定“非空”。由于查询只涉及到一张表,所以效率较高。

2 )每个子类一张附表:

意思是每个类均会有一张表,但是它不是完整的表,因为它的一些字段还在父类的表中。即是说:公共字段放在父表中,子类子类分别放在子类所映射的表中,它们之间采取主外键关联。这样解决了上面的“不能设定为空”的缺限。接上面只需修改 Employee.hbm.xml 配置文件,修改后的内容如下:

< class name = "Employee" >

       < id name = "id" >

           < generator class = "native" />

       </ id >

       < property name = "name" ></ property >

       < many-to-one name = "depart" column = "depart_id" />

       < joined-subclass name = "Sale" >

           < key column = "sale_id" />

           < property name = "signSale" />

       </ joined-subclass >

      < joined-subclass name = "Skill" >

           < key column = "skill_id" />

           < property name = "signSkill" />

       </ joined-subclass >

</ class >

配置文件说明:当每个子类都会有一张表,在class 子元素下设定了 <joined-subclass> 元素, 它的意思就是专门指定为子类映射成一张表,特别要说明的是我们为每个子类都配置了<key> 元素,它的作用就是作为外键关联employee, 它的值也是参照class 元素的id 来生成。执行后的表内容如下:

skill

+----------+-----------+

| skill_id | signSkill |

+----------+-----------+

|        3 | skillName |

+----------+-----------+

sale

+---------+----------+

| sale_id | signSale |

+---------+----------+

|       2 | saleName |

+---------+----------+

 

employee

+----+---------------+-----------+

| id | name          | depart_id |

+----+---------------+-----------+

|  1 | empName       |         1 |

|  2 | saleEmployee  |         1 |

|  3 | skillEmployee |         1 |

+----+---------------+-----------+

执行后请留意hibernate 产生的sql 语句。

3 )联合使用表:

意思是同时使用(1 )(2 )的形式,主要目的是为了能使用鉴别类型,但同时也能“设定为非空”。同样只需要修改配置文件,修改后的内容如下:
< class name = "Employee" discriminator-value = "0" >

       < id name = "id" >

           < generator class = "native" />

       </ id >

       < discriminator column = "sign" type = "string" />

       < property name = "name" ></ property >

       < many-to-one name = "depart" column = "depart_id" />

        < subclass name = "Sale" discriminator-value = "1" >

           < property name = "signSale" ></ property >

       </ subclass >

       < subclass name = "Skill" discriminator-value = "2" >

           < join table = "skill" >

              < key column = "skill_id" />

              < property name = "signSkill" />

           </ join >

       </ subclass >

</ class >
观察上面的配置文件,不难发现我们对Sale 采取了(1 )方式,对Skill 采取了(3 )方式。同样请留意hibernatesql 语句和执行后的表。特别要说明的是:在进行操作时,由于hibernate 在进行操作时不能删除前面的相关联的表(主要是和第一种继承关系“共享一张表”时发生冲突,因为在建立的三张表,在未删除sale/skill 表时,是不能来删除employee 表,因为employee 表中的主键被另两张表关联了),所以需要手工删除表或者是直接删除数据库再建数据库。 补充:借助此例我们来看看 discriminator-value 的默认情况:如果我们在 < subclass name = "Skill" discriminator-value = "2" > 中不写 discriminator-value 配置,再来执行发现一切正常,查表发现它的sign 值为“com.asm.hibernate.domain.Skill ”即完整的类名,属于字串,所以执行正常。但当我们把 <discriminator> 中的type 属性改成int 时将会报错,因为默认不写是以完整的类名字串来作为标识,而设定typeint, 所以将不能匹配类型。

4 )每个具体类一张完整表:

与第(2 )种情况相比,它的主要特点就是为每个具体类建立一张表,表的字段对应子类本身的属性,同样也包括父类的所有属性。 同样只需要修改配置文件如下:

< class name = "Employee" >

       < id name = "id" >

           < generator class = "hilo" />

       </ id >

       < property name = "name" ></ property >

       < many-to-one name = "depart" column = "depart_id" />

       < union-subclass name = "Sale" >

           < property name = "signSale" />

        </union-subclass>
        <union-subclass name="Skill">
            <property name="signSkill" />
        </union-subclass>
    </class>
注意:id生成器选择了“hilo”,由于我们为每个具体类都映射了一张表,所以id不能只在每张表中递增,如果只在每张表中递增,这里的三张表中的id将会出现重复,这样我们在采用多态查询,将会查出多种结果,我们应让id是唯一的,所以采取了hilo的方式来生成id,它能保证id是全局性的递增生成,这样每张表中的id均不会重复。 同样请注意,可能需要删除某些表或者是删库建库才能执行测试类。 执行完成后,需留意hibernate产生的sql语句和表的结构内容。 补充说明:在使用这种方法时,如果父类为抽象类也是可行得,我们可以在<class>元素中配置abstract=”true”来说明父类为抽象类,自然就不会为其建表。
总结:在上面的继续关系中我们多次用到了删库建库,在执行测试类时,如果出现sql不能更新或者sql相关的错误,则不防尝试此方法。另外在学继承关系时,除了注意配置文件外,更应注意hibernate产生的sql语句以及执行后产生表的情况。通常我们建议表的数目不要超过类的数目。

分享到:
评论

相关推荐

    hibernate入门

    Hibernate 是一个开源的对象关系映射(ORM)框架,它允许Java开发者在Java应用程序中操作数据库,而无需直接编写SQL语句。ORM框架如Hibernate简化了数据持久化的过程,通过将对象模型与数据库模型进行映射,实现了...

    hibernate入门与概述

    【hibernate入门与概述】 Hibernate 是一个流行的Java对象关系映射(ORM)框架,它允许开发者使用面向对象的方式操作数据库,将Java类与数据库表进行映射,从而简化数据库操作。本文主要涵盖以下几个核心概念: 1....

    MyEclipse+Hibernate快速入门中文版

    - 多态关联:能够处理继承关系中的关联问题。 - SQL查询的自定义:允许开发者编写自己的SQL语句进行复杂查询。 Hibernate的核心配置文件为`hibernate.cfg.xml`,在这个文件中可以配置数据库连接信息、使用的数据库...

    Hibernate入门到精通

    《Hibernate入门到精通》 Hibernate 是一款强大的Java持久层框架,它简化了数据库与Java对象之间的交互,使得开发者能够更加专注于业务逻辑的实现。本文将深入探讨Hibernate的基础知识,包括各种关联映射、继承映射...

    hibernate入门课件--introduce hibernate.pdf

    ### Hibernate入门知识点详解 #### 一、概述 在IT领域,特别是软件开发中,数据库与对象之间的交互一直是项目中的核心部分。为了更好地处理这一过程,众多框架和技术应运而生,其中Hibernate作为一款优秀的对象...

    Hibernate 入门

    - 本篇概述了Hibernate入门所需了解的关键知识点,从环境搭建到核心概念再到实际应用场景均有涉及。学习Hibernate不仅可以提升开发效率,还能帮助开发者更好地理解ORM框架的工作原理和技术优势。通过上述内容的学习...

    HIBERNATE入门

    【HIBERNATE入门】是针对Java开发者的培训资料,主要介绍了如何入门并掌握Hibernate这一流行的O/R Mapping框架。Hibernate是一种用于简化Java应用程序与数据库交互的开源库,它通过对象关系映射(O/R Mapping)技术...

    Hibernate快速入门教程

    最后,Hibernate提供了强大的高级特性,比如继承映射、多态关联映射、组件映射等。这些高级特性使得Hibernate能够处理更复杂的业务场景和对象模型,极大的扩展了ORM的能力。 通过以上的知识点梳理,我们可以看到...

    hibernate基础 入门/ hibernate basic

    2. **支持常见的面向对象编程概念**:Hibernate支持面向对象编程中常见的概念,如关联(association)、继承(inheritance)、多态(polymorphism)、组合(composition)等。 3. **高性能的对象缓存机制**:...

    Hibernate_3.2.0_符合Java习惯的关系数据库持久化

    1. Hibernate入门 1.1. 前言 1.2. 第一部分 - 第一个Hibernate应用程序 1.2.1. 第一个class 1.2.2. 映射文件 1.2.3. Hibernate配置 1.2.4. 用Ant构建 1.2.5. 启动和辅助类 1.2.6. 加载并存储对象 1.3. 第...

    hibernate入门文档

    ### Hibernate入门精要 #### Hibernate概述 Hibernate是一个开放源代码的ORM(Object Relational Mapping)框架,用于Java应用中实现数据库操作。它提供了一个高级的数据持久化层,简化了对象和关系型数据库之间的...

    MyEclipse_Hibernate_快速入门_中文版.

    - 多态关联:能够处理继承层次中的对象关联。 - 自定义SQL查询:允许使用原生SQL进行更复杂的查询操作。 #### 5. 创建HibernateDemo项目 - **步骤1**: 创建HibernateDemo Java Project。 - **步骤2**: 添加...

    Hibernate快速入门

    在“Hibernate快速入门”中,我们将深入理解Hibernate的核心概念,包括实体映射、对象关系映射(ORM)、Session管理以及查询语言。 首先,Hibernate通过ORM机制将Java对象与数据库表进行映射,使得我们可以在Java...

    hibernate_reference使用指南全

    Hibernate 是一个开源的对象关系映射 (ORM) 框架,它为 Java 应用程序提供了一种将 Java 对象映射到关系型数据库表中的机制。本章节将详细介绍如何从零开始创建一个简单的 Hibernate 应用程序。 **1.2 第一个 ...

    Hibernate3.1_学习源码

    案例目录: 01 01Hibernate_Handwork : 手工配置使用Hibernate,其中详细标了...07 07Hibernate_Mapping : Hibernate中的数据关联技术,是一个重点又是一个难点,演示了 一对一、多对一、一对多、多对多等几种情况。

    Hibernate入门

    【Hibernate入门】 Hibernate 是一款流行的 Java ORM (Object-Relational Mapping) 框架,用于简化Java应用程序与关系数据库之间的交互。ORM 技术旨在解决编程语言对象模型与关系数据库之间的范例不匹配问题,使得...

    hibernate_3.2_官方中文参考手册.

    Hibernate入门与核心概念 Hibernate是一款开源的对象关系映射(ORM)框架,它能够将面向对象的Java应用程序与关系型数据库进行桥接,使得开发者可以使用面向对象的方式来进行数据库操作,而无需直接编写SQL语句。...

    hibernate3.6.3中文api

    - **继承策略**:详细讨论了各种继承模式及其在 Hibernate 中的应用。 #### 五、配置与优化 - **配置属性**:列举了多种配置选项,如 SQL 方言的选择、外连接抓取、二进制流处理、二级缓存配置等。 - **日志配置**...

    hibernate 3.2中文手册 中文文档

    #### 第1章:Hibernate入门 - **1.1 前言** - 介绍了Hibernate的基本概念及其重要性。 - 强调了学习Hibernate对于Java开发人员的重要性。 - **1.2 第一部分—第一个Hibernate应用程序** - **1.2.1 第一个Class*...

Global site tag (gtag.js) - Google Analytics