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

 更新时间:2015年12月28日 17:37:16   作者:cxshun  
这篇文章主要介绍了Java的Hibernate框架中的继承关系设计,Hibernate是Java的SSH三大web开发框架之一,需要的朋友可以参考下

这次我们来说一下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中指定。
 这样当我们运行后生成的表就如下图:

20151228173509059.png (586×227)

两个子类对应的表只有我们通过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有值。
 这样我们就只需要用到一个表,就表明了它们几个类的关系了,注意,这种方式对有过多子类的情况下,并不好,它会使主表的字段过多,会造成一定的设计上的不便。

相关文章

  • SpringBoot整合Hbase的实现示例

    SpringBoot整合Hbase的实现示例

    这篇文章主要介绍了SpringBoot整合Hbase的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-12-12
  • logback的addtivity属性定义源码解读

    logback的addtivity属性定义源码解读

    这篇文章主要为大家介绍了logback的addtivity属性定义源码解读,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-11-11
  • java.io.IOException:你的主机中的软件中止了一个已建立的连接踩坑实战

    java.io.IOException:你的主机中的软件中止了一个已建立的连接踩坑实战

    最近在工作中遇到了个问题,分享给同样遇到问题的同学,这篇文章主要给大家介绍了关于java.io.IOException:你的主机中的软件中止了一个已建立的连接的踩坑实战记录,需要的朋友可以参考下
    2023-03-03
  • Java结构型设计模式之组合模式Composite Pattern详解

    Java结构型设计模式之组合模式Composite Pattern详解

    组合模式,又叫部分整体模式,它创建了对象组的数据结构组合模式使得用户对单个对象和组合对象的访问具有一致性。本文将通过示例为大家详细介绍一下组合模式,需要的可以参考一下
    2022-11-11
  • Springboot整合ActiveMQ实现消息队列的过程浅析

    Springboot整合ActiveMQ实现消息队列的过程浅析

    昨天仔细研究了activeMQ消息队列,也遇到了些坑,下面这篇文章主要给大家介绍了关于SpringBoot整合ActiveMQ的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-02-02
  • Guava Retryer实现接口重试的示例

    Guava Retryer实现接口重试的示例

    本文主要介绍了Guava Retryer实现接口重试的示例,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-12-12
  • Java对象转换的方案分享

    Java对象转换的方案分享

    这篇文章主要介绍了Java对象转换的方案分享,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的朋友可以参考一下
    2022-09-09
  • Java的Spring框架中DAO数据访问对象的使用示例

    Java的Spring框架中DAO数据访问对象的使用示例

    这篇文章主要介绍了Java的Spring框架中DAO数据访问对象的使用示例,分为在Spring中DOA与JDBC以及与Hibernate的配合使用两种情况来进行演示,需要的朋友可以参考下
    2016-03-03
  • Mybatis常用注解中的SQL注入实例详解

    Mybatis常用注解中的SQL注入实例详解

    MyBatis是一款优秀的持久层框架,它支持定制化 SQL(灵活)、存储过程(PLSQL模块化的组件,数据库的一部分)以及高级映射(表映射为Bean也可以将Bean映射为表),下面这篇文章主要给大家介绍了关于Mybatis常用注解中的SQL注入的相关资料,需要的朋友可以参考下
    2022-02-02
  • Java如何跳出当前多重循环你知道吗

    Java如何跳出当前多重循环你知道吗

    这篇文章主要为大家介绍了Java跳出当前多重循环,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-01-01

最新评论