union-subclass映射与joined-subclass映射非常相似,子类增加的属性也可以有非空约束------即父类实例的数据保存在父表中,而子类实例的数据则保存在子表中。与joined-subclass不同的是,子类实例的数据仅保存在子类表中,没有在父类表中有任何记录。在这种映射策略下,子类表的字段会比父类表的字段要多,因为子类表的字段等于父类属性加子类增加属性的总和。
Person.hbm.xml :
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="db.domain">
<class name="Person" table="persons">
<id name="id" column="id">
<generator class="hilo"/>
</id>
<property name="name" type="string"/>
<property name="gender" type="string"/>
<component name="address">
<property name="detail"/>
<property name="zip"/>
<property name="country"/>
</component>
<union-subclass name="Employee" table="employee">
<property name="title"/>
<property name="salary"/>
<set name="customers" inverse="true">
<key column="employee_id"/>
<one-to-many class="Customer"/>
</set>
<many-to-one name="manager" column="manager_id"/>
<union-subclass name="Manager" table="manager">
<property name="department"/>
<set name="employees" inverse="true">
<key column="manager_id"/>
<one-to-many class="Employee"/>
</set>
</union-subclass>
</union-subclass>
<union-subclass name="Customer" table="customer">
<property name="comments" not-null="true"/>
<many-to-one name="employee" column="employee_id" not-null="true"/>
</union-subclass>
</class>
</hibernate-mapping>
使用union-subclass映射策略,非常简洁,既不需要使用辨别者列,也不需要使用key子元素映射共有主键列,只要使用union-subclass映射子类即可。这种策略方便使用,李刚很喜欢。
运行PersonManager.java ,查看数据库:
这种方式的局限在于,如果一个属性在超类中做了映射,其字段名必须与所有子类表中定义的相同。除此之外,使用union-subclass映射策略是不可使用identity主键生成策略的,因为同一类继承层次中所有实体类必须使用同一个主键种子(即多个持久化实体对应的记录的主键是连续的)。受此影响,也不应该使用native主键生成策略,因为native会根据数据库来选择使用identity或sequence策略。
在这种映射策略下,不同持久化类实例保存在不同的表中,不会出现加载一个实例内容时需要跨越多个表取数据的情况。但是在这种映射策略下,如果执行多态查询,也需要跨越多表进行查询。例如查询满足某个条件的Person实例,Hibernate将会从persons表中查询数据,也会从Person的所有子类对应的表中查询数据,然后对这些结果数据进行union运算------这也是这种映射策略被称为union-subclass的原因。