浅析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有值。
这样我们就只需要用到一个表,就表明了它们几个类的关系了,注意,这种方式对有过多子类的情况下,并不好,它会使主表的字段过多,会造成一定的设计上的不便。
推荐阅读
-
浅析Java的Hibernate框架中的继承关系设计
-
在Java的Hibernate框架中对数据库数据进行查询操作
-
详解Java的Hibernate框架中的缓存与原生SQL语句的使用
-
详解Java的Hibernate框架中的set映射集与SortedSet映射
-
举例讲解Java的Hibernate框架中的多对一和一对多映射
-
详解Java的Hibernate框架中的注解与缓存
-
简介Java的Hibernate框架中的Session和持久化类
-
详解Java的Hibernate框架中的List映射表与Bag映射
-
浅析Java设计模式编程中的单例模式和简单工厂模式
-
Java的Spring框架中bean的继承与内部bean的注入