我們繼續學習Hibenrnate映射中的繼承映射。繼承的概念大家應該都很熟悉,而它在Hibenrnate中的實現方式總共
有三種:
1.每棵類繼承樹一張表。
2.每個類一張表。
3.每個具體類一張表。
每棵類繼承樹一張表
映射文件:
<span style="font-size:18px;"><?xmlversion="1.0"?>
<!DOCTYPEhibernate-mapping PUBLIC
"-//Hibernate/HibernateMapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mappingpackage="com.wyj.hibernate">
<!-- lazy值爲false時,支持多態查詢 -->
<classname="Animal" table="t_animal" lazy="false">
<idname="id">
<generatorclass="native"/>
</id>
<discriminatorcolumn="type" type="string"/>
<propertyname="name"/>
<propertyname="sex"/>
<subclassname="Pig" discriminator-value="P">
<propertyname="weight"></property>
</subclass>
<subclassname="Brid" discriminator-value="B">
<propertyname="height"></property>
</subclass>
</class>
</hibernate-mapping></span>
說明:
因爲類繼承樹肯定是對應多個類,要把多個類的信息存放在一張表中,必須有某種機制來區分哪些記錄是屬於哪
個類的。這種機制就是,在表中添加一個字段,用這個字段的值來進行區分。其實到關係表中就是添加冗餘字段。
一、用hibernate實現這種策略的時候,有如下步驟:
1、父類用普通的<class>標籤定義
2、在父類中定義一個discriminator,即指定這個區分的字段的名稱和類型。如:<discriminator column=”XXX”
type=”string”/>
3、子類使用<subclass>標籤定義,在定義subclass的時候,需要注意如下幾點:
二、Subclass標籤
1、name屬性是子類的全路徑名。
2.在Subclass標籤中,用discriminator-value屬性來標明本子類的discriminator字段(用來區分不同類的字段)
的值。
3.Subclass既可以被class標籤所包含(這種包含關係正是表明了類之間的繼承關係),也可以與class標籤平
行。當subclass標籤的定義與class標籤平行的時候,需要在subclass標籤中,添加extends屬性,裏面的值是
父類的全路徑名稱。
4.子類的其它屬性,像普通類一樣,定義在subclass標籤的內部。
5.關於鑑別值在存儲的時候hibernate會自動存儲,在加載的時候會根據鑑別值取得相關的對象。
每個類一張表
映射文件:
<span style="font-size:18px;"><?xmlversion="1.0"?>
<!DOCTYPEhibernate-mapping PUBLIC
"-//Hibernate/HibernateMapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mappingpackage="com.wyj.hibernate">
<classname="Animal" table="t_animal" lazy="false">
<idname="id">
<generatorclass="native"/>
</id>
<propertyname="name"/>
<propertyname="sex"/>
<joined-subclassname="Pig" table="t_pig">
<key column="pid"/>
<property name="weight"/>
</joined-subclass>
<joined-subclassname="Brid" table="t_brid">
<key column="bid"/>
<property name="height"/>
</joined-subclass>
</class>
</hibernate-mapping></span>
說明:
這種策略是使用joined-subclass標籤來定義子類的。父類、子類,每個類都對應一張數據庫表。在父類對應的數
據庫表中,實際上會存儲所有的記錄,包括父類和子類的記錄;在子類對應的數據庫表中,這個表只定義了子類中所
特有的屬性映射的字段。子類與父類,通過相同的主鍵值來關聯。
實現這種策略的時候,有如下步驟:
1、父類用普通的<class>標籤定義即可
2、父類不再需要定義discriminator字段
3、子類用<joined-subclass>標籤定義,在定義joined-subclass的時候,需要注意如下幾點:
(1)Joined-subclass標籤的name屬性是子類的全路徑名
(2)Joined-subclass標籤需要包含一個key標籤,這個標籤指定了子類和父類之間是通過哪個字段來關聯
的。如:<key column=”PARENT_KEY_ID”/>,這裏的column,實際上就是父類的主鍵對應的映射字段
名稱。
(3)Joined-subclass標籤,既可以被class標籤所包含(這種包含關係正是表明了類之間的繼承關係),也
可以與class標籤平行。當Joined-subclass標籤的定義與class標籤平行的時候,需要在Joined-
subclass標籤中,添加extends屬性,裏面的值是父類的全路徑名稱。
(4)子類的其它屬性,像普通類一樣,定義在joined-subclass標籤的內部。
每個具體類一張表
映射文件:
<span style="font-size:18px;"><?xmlversion="1.0"?>
<!DOCTYPEhibernate-mapping PUBLIC
"-//Hibernate/HibernateMapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mappingpackage="com.wyj.hibernate">
<classname="Animal" table="t_animal" lazy="false"abstract="true">
<idname="id">
<generatorclass="assigned"/>
</id>
<propertyname="name"/>
<propertyname="sex"/>
<union-subclassname="Pig" table="t_pig">
<property name="weight"/>
</union-subclass>
<union-subclassname="Brid" table="t_brid">
<property name="height"/>
</union-subclass>
</class>
</hibernate-mapping></span>
說明:
這種策略是使用union-subclass標籤來定義子類的。每個子類對應一張表,而且這個表的信息是完備的,即包含
了所有從父類繼承下來的屬性映射的字段(這就是它跟joined-subclass的不同之處,joined-subclass定義的子類的
表,只包含子類特有屬性映射的字段)。
實現這種策略的時候,有如下步驟:
1、父類用普通<class>標籤定義即可
2、子類用<union-subclass>標籤定義,在定義union-subclass的時候,需要注意如下幾點:
(1)Union-subclass標籤不再需要包含key標籤(與joined-subclass不同)
(2)Union-subclass標籤,既可以被class標籤所包含(這種包含關係正是表明了類之間的繼承關係),
也可以與class標籤平行。當Union-subclass標籤的定義與class標籤平行的時候,需要在Union-
subclass標籤中,添加extends屬性,裏面的值是父類的全路徑名稱。
(3)子類的其它屬性,像普通類一樣,定義在Union-subclass標籤的內部。這個時候,雖然在union-
subclass裏面定義的只有子類的屬性,但是因爲它繼承了父類,所以,不需要定義其它的屬性,在
映射到數據庫表的時候,依然包含了父類的所有屬性的映射字段。
注意:在保存對象的時候id不能重複(不能使用數據庫的自增方式生成主鍵)
總結:
繼承映射的學習主要是對Subclass標籤、joined-subclass標籤和union-subclass標籤的理解。我們可以根據實際
情況來選擇不同的實現方式,當數據量比較少,表結構簡單時,可以採用每棵類繼承樹一張表來實現,當繼承關係復
雜時,可以採用其他兩種方式來實現,但是每個具體類一張表這種方式映射出的數據庫更符合我們平時的習慣,一個
實體對應一個表。