当我们的两个或多个实体类是相似的,或者说他们有很多相同的属性时,我们通常抽象出父类,子类继承的方式去实现,以面相对象的特点对实体类进行抽象,封装和继承。我们之前学了Hibernate的关系映射,让我们可以直接面向对象开发,而不是面向数据库,如何将继承父类的这种关系反映到数据库中呢,我们来看Hibernate提供的三种映射策略:
映射1-每棵类继承树对应一张表
映射2-每个类一张表
映射3-每个具体类一张表
我们下面以一个简单的例子说明
首先看我们的类图
Pig和Bird都继承自Animal,通过这个简单的关系来看继承映射:
映射1-每棵类继承树对应一张表
每棵类继承树对应一张表,上面的类图就是一棵类继承树吧,只映射出一张表,如下:
这是如何做到的,直接看我们的映射文件:
<hibernate-mapping package="com.bjpowernode.hibernate">
<class name="Animal" table="t_animal" lazy="false">
<id name="id">
<generator class="native"/>
</id>
<discriminator column="type" type="string"/>
<property name="name"/>
<property name="sex"/>
<subclass name="Pig" discriminator-value="P">
<property name="weight"/>
</subclass>
<subclass name="Bird" discriminator-value="B">
<property name="height"/>
</subclass>
</class>
</hibernate-mapping>
在我们的表中有一个Type字段,他不对应任何一个实体的属性。Hibernate用这个列自动初始化对应的类并相应进行填充。
<discriminator column="type"type="string"/>,通过discriminator 设置type鉴别字段,在子类中使用discriminator-value属性标识每条数据所对应的类。
映射2-每个类一张表
这种策略是使用joined-subclass标签来定义子类的。父类、子类,每个类都对应一张数据库表。
在父类对应的数据库表中,实际上会存储所有的记录,包括父类和子类的记录;在子类对应的数据库表中,这个表只定义了子类中所特有的属性映射的字段。子类与父类,通过相同的主键值来关联。如下图:
下面具体的用法:
<hibernate-mapping package="com.bjpowernode.hibernate">
<class name="Animal" table="t_animal">
<id name="id">
<generator class="native"/>
</id>
<property name="name"/>
<property name="sex"/>
<joined-subclass name="Pig" table="t_pig">
<key column="pid"/>
<property name="weight"/>
</joined-subclass>
<joined-subclass name="Bird" table="t_bird">
<key column="bid"/>
<property name="height"/>
</joined-subclass>
</class>
</hibernate-mapping>
需要注意的:
(1)父类用普通的<class>标签定义即可
(2)父类不再需要定义discriminator字段
(3)子类用<joined-subclass>标签定义,在定义joined-subclass的时候,需要注意如下几点:
Joined-subclass标签的name属性是子类的全路径名
Joined-subclass标签需要包含一个key标签,这个标签指定了子类和父类之间是通过哪个字段来关联的。
此策略特点:
Hibernate每次初始化和填充对象时,会读取多个表,如果继承层次多了,性能就会收到影响。
映射3-每个具体类一张表
每个具体类一个表,抽象类没有表。
这种策略使用union-subclass标签来定义子类的。每个子类对应一张表,而且这个表包含了所有从父类继承下来的属性映射的字段(这就是它跟joined-subclass的不同之处,joined-subclass定义的子类的表,只包含子类特有属性映射的字段)。
下面具体的用法:
<hibernate-mapping package="com.bjpowernode.hibernate">
<class name="Animal" table="t_animal" abstract="true">
<id name="id">
<generator class="assigned"/>
</id>
<property name="name"/>
<property name="sex"/>
<union-subclass name="Pig" table="t_pig">
<property name="weight"/>
</union-subclass>
<union-subclass name="Bird" table="t_bird">
<property name="height"/>
</union-subclass>
</class>
</hibernate-mapping>
需要注意的:
(1) Union-subclass标签不再需要包含key标签(与joined-subclass不同)
(2) Union-subclass标签,可以包含在class标签里,说明是继承的class类,也可以不包含在class标签中,但需要添加extends属性,值里是父类的全路径。
(3) 此策略,子类会继承父类的所有属性,而子类的其它属性,可以定义在Union-subclass标签的内部。
总结
Hibernate 映射策略的三个策略自己的优势与不足,根据不同的情况去使用,但不管哪种策略,在实现的整个过程中,我们的业务对象与Hibernate持续性框架之间没有联系,三个策略完全通过Hibernate的xml配置管理,很体现了Hibernate的灵活性,值得我们学习。