hibernate映射继承关系

在数据库设计中,会遇到继承关系的表的设计,一般来说有以下三种映射方法:(假设有HOURLY_EMPLOYEES和SALARIED_EMPLOYEES两个子类,父类EMPLOYEES)

1.把每个子具体类都映射一张表,即只有HourlyEmployee.hbm.xml和SalariedEmployee.hbm.xml两个配置文件。

好处:简单方便

坏处:不支持多态查询

例如,为了检索所有employee对象,必须分别检索所有的HourlyEmployee实例和SalariedEmployee实例,然后把它们合并到同一个集合中。

2.只为根类EMPLOYEES创建一张表,将公共属性和子类特有属性全部引入,并且加一个类型字段EMPLOYEE_TYPE,用于区分Employee的具体类型。

好处:支持多态查询

检索所有employee对象时,如果EMPLOYEE_TYPE字段取值为"HE",就创建HoulyEmployee实例,如果EMPLOYEE_TYPE字段取值为"SE",就创建SalariedEmployee实例。

坏处:配置文件相应变得复杂:如

 
Employee.hbm.xml
  
  <hibernate-mapping >
  <class name="mypack.Employee" table="EMPLOYEES">
  <id name="id" type="long" column="ID">
  <generator class="increment"/>
  </id>
  <discriminator column="EMPLOYEE_TYPE" type="string" />
  <property name="name" type="string" column="NAME" />
  <subclass name="mypack.HourlyEmployee" discriminator-value="HE" >
  <property name="rate" column="RATE" type="double" />
  </subclass>
  
  <subclass name="mypack.SalariedEmployee" discriminator-value="SE" >
  <property name="salary" column="SALARY" type="double" />
  </subclass>
  
  </class>
  
  </hibernate-mapping>

另外,如果公共属性和特有属性比较多时,字段会变得很多,表显得臃肿。

3.为根类和子类都创建映射表,即需要创建EMPLOYEES、HOURLY_EMPLOYEES和SALARIED_EMPLOYEES表。EMPLOYEES表仅包含和Employee类的属性对应的字段(子类共有的属性),HOURLY_EMPLOYEES和SALARIED_EMPLOYEES表也仅包含其类中有的属性对应的字段(特有的属性),另外HOURLY_EMPLOYEES和SALARIED_EMPLOYEES表中都以EMPLOYEES表中的EMPLOYEE_ID字段(主键)作为主键,同时也作为外键。

好处:同样支持多态查询,

    
Employee.hbm.xml
  
  <hibernate-mapping >
  
  <class name="mypack.Employee" table="EMPLOYEES">
  <id name="id" type="long" column="ID">
  <generator class="increment"/>
  </id>
  <property name="name" type="string" column="NAME" />
  
  <many-to-one
  name="company"
  column="COMPANY_ID"
  class="mypack.Company"
  />
  
  <joined-subclass name="mypack.HourlyEmployee"  table="HOURLY_EMPLOYEES" >
  <key column="EMPLOYEE_ID" />
  <property name="rate" column="RATE" type="double" />
  </joined-subclass>
  
  <joined-subclass name="mypack.SalariedEmployee" table="SALARIED_EMPLOYEES" >
  <key column="EMPLOYEE_ID" />
  <property name="salary" column="SALARY" type="double" />
  </joined-subclass>
  
  </class>
  </hibernate-mapping>

通过左外连接获得子类对象的所有属性,如果HE表的EMPLOYEE_ID字段不为null,就创建HoulyEmployee实例,如果SE表的EMPLOYEE_ID字段不为null,就创建SalariedEmployee实例。

当保存一个子类对象时,需要发出两条sql语句分别向子表和EMPLOYEES表中插入记录。(session的save()方法能判断实际引用的子类类型,从而插入记录到不同的子表)那么

总的来说,具体怎么设计,还是要看情况....

相关推荐