上篇文章《Eclipse快速上手Hibernate--4. 繼承映射(1) 》中已經談了每個類層次結構一個表(table per class hIErarchy)的策略,這篇文章主要說的是每個子類一個表(table per subclass)的策略。一些重復的部分這裡就不說了,請參考上篇文章。 1. 創建項目 · 繼續沿用上篇文章中所建的Java項目:InheritanceMapping。 2. 編寫類文件 · 新建一個類,包名:javamxj.inheritance.two,類名:Animal。然後在生成的代碼中添加變量,再利用“生成 Getter 和 Setter”,具體方式同《Eclipse快速上手Hibernate--1. 入門實例 》文章中的編輯User.Java的方式一樣。· 這個類是父類,只是生成一個簡單的表。
Vehicle.Java
/* * Hibernate - 繼承映射(每個子類一個表) * 創建日期 2005-4-9 * @author javamxj(分享java快樂) * @link Blog: htpp://Javamxj.mblogger.cn * htpp://blog.csdn.Net/javamxj/ */package Javamxj.inheritance.two;/** * @hibernate.class */public class Vehicle {private Long id;private String name;/** * @hibernate.id * column="ID" * generator-class="hilo" * unsaved-value="null" */public Long getId() {return id;}public void setId(Long id) {this.id = id;}/** * @hibernate.property * length = "24" */public String getName() {return name;}public void setName(String name) {this.name = name;}}
· 子類Car.javaCar.Java
package Javamxj.inheritance.two;/** * @hibernate.joined-subclass * @hibernate.joined-subclass-key * column="id" */public class Car extends Vehicle {private String seat;/** * @hibernate.property * column = "載客" * length = "24" */public String getSeat() {return seat;}public void setSeat(String seat) {this.seat = seat;}}
· 子類Truck.javaTruck.Java
package Javamxj.inheritance.two;/** * @hibernate.joined-subclass * @hibernate.joined-subclass-key * column="id" */public class Truck extends Vehicle {private String load;/** * @hibernate.property * column = "載重" * length = "24" */public String getLoad() {return load;}public void setLoad(String load) {this.load = load;}}
· 這兩個子類都很簡單,注意添加的hibernate.joined-subclass的標記。 · 好了,這時整個項目的結構如下: 3. 運行任務 · 雙擊“generate-hbm”任務,會發現在包中多了一個Vehicle.hbm.XML文件。如果沒有,按F5鍵刷新一下(這裡建議打開Eclipse的“首選項”對話框,在“工作台”中勾選“自動刷新工作空間”和“在構建之前自動保存”這兩項,這樣以後不用每次都刷新了)。
Vehicle.hbm.XML
· 重點是看看“joined-subclass”標簽。 · 同時在hibernate.cfg.XML文件中會自動添加一個映射文件信息:
Demo.Java
/* * Hibernate - 繼承映射(每個子類一個表) * 創建日期 2005-4-9 * @author javamxj(分享java快樂) * @link Blog: htpp://Javamxj.mblogger.cn * htpp://blog.csdn.Net/javamxj/ */package javamxj.inheritance.two;import java.util.Iterator;import Java.util.List;import net.sf.hibernate.HibernateException;import net.sf.hibernate.Session;import net.sf.hibernate.SessionFactory;import net.sf.hibernate.Transaction;import net.sf.hibernate.cfg.Configuration;public class Demo {public static void main(String[] args) {try {new Demo();} catch (HibernateException he) {he.printStackTrace();}}public Demo() throws HibernateException {SessionFactory sf = new Configuration().configure().buildSessionFactory();Session sess = sf.openSession();Transaction tx = null;try {tx = sess.beginTransaction();Car car = new Car();car.setName("奇瑞QQ");car.setSeat("4人");sess.save(car);Truck truck = new Truck();truck.setName("一汽解放");truck.setLoad("10噸");sess.save(truck);tx.commit();} catch (HibernateException e) {if (tx != null)tx.rollback();throw e;} finally {sess.close();}sess = sf.openSession();tx = null;try {tx = sess.beginTransaction();List pets = sess.find("from " + Vehicle.class.getName());for (Iterator it = pets.iterator(); it.hasNext();) {Vehicle vehicle = (Vehicle) it.next();System.out.println("車型 " + vehicle.getName()+ " its class is: " + vehicle.getClass().getName());}tx.commit();} catch (HibernateException e) {if (tx != null)tx.rollback();throw e;} finally {sess.close();}}}
· 運行這個類,控制台輸出如下: · 同時,數據表中生成如下數據: 小結: ● 優點:· 與面向對象的概念的一致性最好。對多態的支持最好,對於對象所可能充當的角色僅需要在相應的表中保存記錄。· 易於修改基類和增加新的類。· 這種映射幾乎有最佳的空間節約性。唯一的容易就是額外的OID,來連接不同的層級層次。 ● 缺點:· 數據庫中存在大量的表。· 訪問數據的時間較長。在讀寫相關的任務上,這種映射方式相當費資源,這個代價還隨著繼承層次的增多而增大。 · 對報表的支持較差,除非定義視圖。 參考:· HIBERNATE - 符合Java習慣的關系數據庫持久化(第8章)· Hibernate 簡化繼承映射· Mapping Objects to Relational Databases: O/R Mapping In Detail· Mapping objects to relational databases 下篇文章會談談每個具體類一個表的策略。