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

作者:cxshun 时间:2021-10-18 03:10:03 

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

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

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

标签:Hibernate,Java
0
投稿

猜你喜欢

  • Java并发编程之线程创建介绍

    2021-11-14 13:55:40
  • Java使用HttpUtils实现发送HTTP请求

    2021-06-11 07:08:39
  • Java使用DOM4j实现读写XML文件的属性和元素

    2021-10-19 00:21:52
  • java IO流 之 输入流 InputString()的使用

    2023-08-22 07:44:31
  • 利用java实现二叉搜索树

    2023-10-25 13:25:07
  • Java运算符从见过到掌握上

    2022-09-08 02:12:43
  • 探讨:android项目开发 统筹兼顾 需要考虑的因素

    2023-08-05 11:21:32
  • C#实现文件与字符串互转的方法详解

    2023-04-26 03:18:12
  • C#绘制饼状图和柱状图的方法

    2023-12-19 15:42:59
  • Java设计模式之职责链模式详解

    2023-01-26 21:16:26
  • java 重定义数组的实现方法(与VB的ReDim相像)

    2022-08-09 23:09:25
  • Java之OutputStreamWriter流案例详解

    2023-11-11 13:03:55
  • Java递归实现斐波那契数列

    2022-05-03 19:00:11
  • springboot2.2 集成 activity6实现请假流程(示例详解)

    2022-09-18 08:54:21
  • Java 多线程同步 锁机制与synchronized深入解析

    2023-12-20 17:55:50
  • 简单捋捋@RequestParam 和 @RequestBody的使用

    2022-11-02 16:19:20
  • Java可变个数形参的方法实例代码

    2023-01-15 18:35:56
  • 使用C#发送Http请求实现模拟登陆实例

    2023-06-22 22:25:07
  • Mybatis一对一延迟加载实现过程解析

    2022-09-07 12:45:43
  • Spring Boot小型项目如何使用异步任务管理器实现不同业务间的解耦

    2022-11-21 07:49:54
  • asp之家 软件编程 m.aspxhome.com