浅析Java的Hibernate框架中的继承关系设计

这次我们来说一下hibernate的层次设计,层次设计也就是实体之间的继承关系的设计。
 也许这样比较抽象,我们直接看例子。
 1)我们先看一下普通的做法
 直接上代码:三个实类如下:

public class TItem implements Serializable{
    //省略Get/Set方法
  private int id;
  private String manufacture;
  private String name;
}
public class TBook extends TItem{
    //省略Get/Set方法
  private int pageCount; 

}
public class TDVD extends TItem{
    //省略Get/Set方法
  private String regionCode;
} 

这里我们需要三个映射文件,内容如下:

<class name="TItem" table="ITEM">
    <id name="id" column="id" type="java.lang.Integer">
      <generator class="native" />
    </id>
    <property name="name" column="name" type="java.lang.String"/>
    <property name="manufacture" column="manufacture" type="java.lang.String"/>
  </class>
<class name="TBook" table="Book">
    <id name="id" column="id" type="java.lang.Integer">
      <generator class="native" />
    </id>
    <property name="name" column="name" type="java.lang.String"/>
    <property name="manufacture" column="manufacture" type="java.lang.String"/>
    <property name="pageCount" column="pageCount" type="java.lang.Integer"/>
  </class>
<class name="TDVD" table="DVD">
    <id name="id" column="id" type="java.lang.Integer">
      <generator class="native" />
    </id>
    <property name="name" column="name" type="java.lang.String"/>
    <property name="manufacture" column="manufacture" type="java.lang.String"/>
    <property name="regionCode" column="regionCode" type="java.lang.String"/>
  </class>

很普通的映射文件,跟以前的没什么区别。
 下面我们直接写一个测试方法:

public void testSelect() { 

    Query query = session.createQuery("from TItem ");
    List list = query.list();
    Iterator iter = list.iterator(); 

    while(iter.hasNext()) {
      System.out.println("Name:"+(((TItem)iter.next()).getName()));
    } 

  }

注意,这里我们是用TItem类,而不是具体的字类,这里它会自动去查找继承于TItem类的子类,查出所有结果。这里涉及到一个多态模式,class标签有属性 polymorphism,它的默认值为implicit,这意味着不需要指定名称就可以查询出结果。如果为explicit则表明需要指定具体的类名时,才可以查出此类的结果。
 2)上个例子中我们用到了三个映射文件,当我们需要修改时,就需要修改三个映射文件,这对于大的项目是很不可行的。而且每个表都有对应的主类的对应字段,这是多余的。所以我们有下面这种方法。
 实体类还是跟1)中的一样。我们把映射文件由三个改为一个,只保留TItem映射文件。但我们需要做相应的修改,现在内容如下:

<class name="TItem" table="ITEM" polymorphism="explicit">
    <id name="id" column="id" type="java.lang.Integer">
      <generator class="native" />
    </id>
    <property name="name" column="name" type="java.lang.String"/>
    <property name="manufacture" column="manufacture" type="java.lang.String"/>
    <joined-subclass name="TBook" table="TBOOK">
      <key column="id" />
      <property name="pageCount" column="pageCount" type="java.lang.Integer" />
    </joined-subclass>
    <joined-subclass name="TDVD" table="TDVD">
      <key column="id"/>
      <property name="regionCode" column="regionCode" type="java.lang.String"/>
    </joined-subclass>
  </class> 

这里,我们只有一个映射文件,但有一个joined-subclass标签,它表明这个类继承于当前类,<key>表明分表的主键,这里分表是指TBOOK和TDVD这两个由子类对应的表。分表中只有字段在property中指定。
 这样当我们运行后生成的表就如下图:

两个子类对应的表只有我们通过property指定的字段。这样就避免了表内有多个字段,使字表只维护其单独字段,当item类进行改变时,也不用过多的进行修改。

3)再来了解另外一种方法实现层次设计,这就是通过在表内置入标志来实现。在hibernate的映射文件中我们通过descriminator标签来进行实现。
 废话不多说,我们直接看例子:
 我们把昨天的TItem的映射文件修改为:

<class name="TItem" table="ITEM" polymorphism="explicit">
    <id name="id" column="id" type="java.lang.Integer">
      <generator class="native" />
    </id>
     <discriminator column="category" type="java.lang.String"/> 

     <property name="name" column="name" type="java.lang.String"/>
     <property name="manufacture" column="manufacture" type="java.lang.String"/>
  </class>

看到中间,我们加入了一个discriminator标签,它表明我们以下的两个subclass通过哪个字段来进行区别。

<subclass name="TBook" discriminator-value="1">
    <property name="pageCount" column="pageCount"/>
   </subclass>
   <subclass name="TDVD" discriminator-value="2" >
    <property name="regionCode" column="regionCode"/>
   </subclass>

我们看到这两段,它指明了当discriminator所指定的field的值为1时,表明它是TBook类,并且pageCount有值;当discriminator所指定的field值为2时,表明它是TDVD类,并且regionCode有值。
 这样我们就只需要用到一个表,就表明了它们几个类的关系了,注意,这种方式对有过多子类的情况下,并不好,它会使主表的字段过多,会造成一定的设计上的不便。

(0)

相关推荐

  • Java的Hibernate框架结合MySQL的入门学习教程

    零.关于Hibernate Hibernate是冬眠的意思,它是指动物的冬眠,但是本文讨论的Hibernate却与冬眠毫无关系,而是接下来要讨论的SSH2框架中的一员.Hibernate是一个开源的项目,它是一个对象关系模型的框架,并且对JDBC进行了非常轻量级的封装,程序员在开发时可以使用对象编程思维进行开发. 下载地址:http://hibernate.org/orm/downloads/ Note:轻量级和重量级的区别,轻量级的框架包较小,并且使用较简单,而且测试容易,开发效率高:重量级框

  • java Hibernate save()与persist()区别

    Hibernate 之所以提供与save()功能几乎完全类似的persist()方法,一方面是为了照顾JPA的用法习惯.另一方面,save()和 persist()方法还有一个区别:使用 save() 方法保存持久化对象时,该方法返回该持久化对象的标识属性值(即对应记录的主键值):但使用 persist() 方法来保存持久化对象时,该方法没有任何返回值.因为 save() 方法需要立即返回持久化对象的标识属性,所以程序执行 save() 会立即将持久化对象对应的数据插入数据库:而 persist

  • Java的Hibernate框架中一对多的单向和双向关联映射

    一.一对多单向关联映射 一对多关系的对象模型在日常生活中也经常看到,就拿学生和班级来说,一个班级里有多个学生,所以班级和学生的关系是一对多的关系,映射到对象模型中,如下图: 对象模型说明了这种一对多的关系是由一的一端来维护的,那么映射成关系模型就是一个班级字段下面会有多个学生,这样就形成了一对多的关系,通过班级能够查询获得学生信息,对应的关系模型如下图: 1.基本配置 有了对象模型接下来就让它们映射为对应的关系代码,在进行关系映射时需要在一的一端添加<one-to-many>标签,另外还需要在

  • Java的Hibernate框架中的继承映射学习教程

    一.继承映射 继承是面向对象很重要的特性,它实现了代码的服用,在关系模型中同样也有继承关系,这种继承关系其实可以看做是一种枚举关系,一种类型中可以枚举出很多子类型,这些子类型和父对象形成了继承关系,能够对其进行枚举的大部分都可以看做是一种继承映射,所以这种枚举关系可以看做是继承映射,例如动物就是一种抽象类,它是其它动物猪.猫等的父类,它们之间就是一种继承关系,如下图: 这种继承映射在转化为关系模型后会生成一张表,那么这张表是如何区分这两种类型的呢?用的是关系字段,需要在表中添加类型字段,使用关键

  • 详解Java的Hibernate框架中的Interceptor和Collection

    Interceptor 讲到Interceptor,相信熟悉struts2的童鞋肯定不会陌生了,struts2可以自定义拦截器进行自己想要的一系列相关的工作.而这里我们说的Interceptor也是差不多相似的功能.  废话不说,直接来代码:  下面这个是MyInterceptor类,它实现了Interceptor接口: public String onPrepareStatement(String arg0) { return arg0; } public boolean onSave(Obj

  • 深入解析Java的Hibernate框架中的持久对象

    一.持久对象生命周期 应用程序在使用Hibernate框架后,创建的持久对象会经历一整套生命周期来完成数据库的操作,其中主要的三个状态分别是瞬态(Transient).持久化(Persistent).脱管(detached).这三种状态的转换是能够在应用程序中控制的,如下图: 为了能清楚的了解这几种状态,这里使用一个实例来查看下这几种状态下对象的不同,下面状态内的代码,具体步骤如下: (1)创建Hibernate_session程序集,并添加像相应的jar包: (2)配置Hibernate,添加

  • 简介Java的Hibernate框架中的Session和持久化类

    Session Session对象用于获取与数据库的物理连接. Session对象是重量轻,设计了一个互动是需要与数据库每次被实例化.持久化对象被保存,并通过一个Session对象中检索. 会话中的对象不应该保持开放很长一段时间,因为他们通常不被线程安全的,他们应该被创建并根据需要摧毁他们.这次会议的主要功能是提供创建,读取和删除操作映射的实体类的实例.实例中可能存在以下三种状态之一在给定时间点: 短暂性: 持久化类的未与会话相关联,并在数据库中没有代表性,没有标识值的新实例被Hibernate

  • Java的Hibernate框架中复合主键映射的创建和使用教程

    复合主键映射需要在映射配置文件中使用<composite-id>标签,该标签是指将一个类指定为相应的复合主键,它的name属性需要指定类文件中定义的属性值,并在该标签中添加<key-property>子标签. Note:想要使用复合映射必须要将复合主键放到一个类中,也就是讲复合主键属性和其它属性分到两个类中,并将复合主键的类实现接口Serializable,该接口隶属于java.io. 复合主键的映射关系的主键是由多个列复合而成的,对应到数据表中相当的简单,如下图: 1.类文件 这

  • Java的Hibernate框架中的组合映射学习教程

    一.组合映射 组合是关联关系的一种特殊情况,是关联关系耦合度最高的一种关系,组合的主对象和子对象拥有相同的生命周期,主对像消亡的话子对象也会消亡.这里使用雇主和用户作为示例,用户和雇主都拥有联系方式属性,如果这里站在对象角度思考的话,常常会把对象模型绘制成为组合的方式,抽象出来一个共同的联系方式类,然后两种人分别包含相应的联系方式对象即可,向应的对象模型时它的对象示例如下图所示: 组合对象模型在生成相应的关系模型后会把对应的子类包含到主表中,所以对应的表结构会将相应的属性生成到对应的表中,相应的

  • Java的Hibernate框架中的双向主键关联与双向外键关联

    一.双向主键关联 双向的主键关联其实是单向一对一主键关联的一种特殊情况,只不过要在关联对象的两端的映射文件中都要进行<one-to-one>的配置,另外还要在主映射的主键一端采用foreign外键关联属性. 这里同样使用Person和IdCard来讨论,一个人对应着一个唯一的身份证,而且一个身份证也唯一映射着一个人,所以这就产生了双向的关联关系,Person的主键同样也是IdCard的主键,分别是主键的同时也是外键,这种关联关系成为双向一对一映射,表现到关系模型中可如下图: 图中的两个表采用了

  • Java的Hibernate框架中的基本映射用法讲解

    Hibernate进行了分类整合发现其实Hibernate分为三大部分:核心对象.映射.HQL,这三大部分开发过程中最常使用,前几篇讨论了核心对象及对象之间的转换方法,接下来讨论Hibernate的映射使用方法.   Hibernate一个重要的功能就是映射,它能够在对象模型和关系模型之间转换,是面向对象编程思想提倡使用的,使用映射程序开发人员只需要关心对象模型中代码的编写.对象和关系数据库之间的映射通常是由XML文档来定义的.这个映射文档被设计为易读的,并且可以手动修改.这种映射关系我总结为下

  • Java Hibernate中使用HQL语句进行数据库查询的要点解析

    一.实体对象查询 实体对象查询是hql查询的基础,作为一种对象查询语言,在查询操作时和sql不同,查询字符串中的内容要使用类名和类的属性名来代替.这种查询方法相对简单,只要有SQL功底,使用hql是很简单的,但是有一些问题需要注意,就是查询获取数据不是目的,需要考虑的是如何编写出高效的查询语句,这才是讨论的重点. 1.N+1问题 (1)什么是N+1问题 在刚听到这个名词时疑惑可能是有的,以前根本就没有听过N+1问题,那么它是指什么呢?N+1指的是一张表中有N条数据,那么在获取这N条数据时会产生N

随机推荐