当前位置导航:炫浪网>>网络学院>>编程开发>>JAVA教程>>Java进阶

精通Hibernate之映射继承关系七(图)


  Company与Employee类之间为一对多多态关联关系,如果继承关系树的根类对应一个表,或者每个类对应一个表,那么就能映射Company类的employees集合。本节介绍如何映射多对一多态关联。如图14-11所示,ClassD与ClassA为多对一多态关联关系。
  
 

  
图14-11 ClassD与ClassA为多对一多态关联关系

  
  ClassA、ClassB和ClassC构成了一棵继承关系树,如果继承关系树的根类对应一个表,或者每个类对应一个表,那么可以按以下方式映射ClassD的a属性:
  
  <many-to-one name="a"
  class="ClassA"
  column="A_ID"
  cascade="save-update" />
  
  假定与ClassD对应的表为TABLE_D,与ClassA对应的表为TABLE_A,在TABLE_D中定义了外键A_ID,它参照TABLE_A表的主键。
  
  ClassD对象的a属性既可以引用ClassB对象,也可以引用ClassC对象,例如:
  
  tx = session.beginTransaction();
  ClassD d=(ClassD)session.get("ClassD",id);
  ClassA a=d.getA();
  if(a instanceof ClassB)
  System.out.println(((ClassB)a).getB1());
  if(a instanceof ClassC)
  System.out.println(((ClassC)a).getC1());
  tx.commit();
  
  以下代码在映射ClassD类的a属性时使用了延迟检索策略:
  
  <many-to-one name="a"
  class="ClassA"
  column="A_ID"
  lazy="true"
  cascade="save-update" />
  
  当Hibernate加载ClassD对象时,它的属性a引用ClassA的代理类实例,在这种情况下,如果对ClassA的代理类实例进行类型转换,会抛出ClassCastException:
  
  ClassA a=d.getA();
  ClassB b=(ClassB)a; //抛出ClassCastException
  
  解决以上问题的一种办法是使用Session.load()方法:
  
  ClassA a=d.getA();
  ClassB b=(ClassB)session.load(ClassB.class,a.getId());
  System.out.println(b.getB1());
  
  当执行Session的load()方法时,Hibernate并不会访问数据库,而是仅仅返回ClassB的代理类实例。这种解决办法的前提条件是必须事先知道ClassD对象实际上和ClassA的哪个子类的对象关联。
  
  解决以上问题的另一种办法是显式使用迫切左外连接检索策略,避免Hibernate创建ClassA的代理类实例,而是直接创建ClassA的子类的实例:
  
  tx = session.beginTransaction();
  ClassD d=(ClassD)session.createCriteria(ClassD.class)
  .add(Expression.eq("id",id))
  .setFetchMode("a",FetchMode.EAGER)
  .uniqueResult();
  ClassA a=d.getA();
  if(a instanceof ClassB)
  System.out.println(((ClassB)a).getB1());
  if(a instanceof ClassC)
  System.out.println(((ClassC)a).getC1());
  tx.commit();
  
  如果继承关系树的具体类对应一个表,为了表达ClassD与ClassA的多态关联,需要在TABLE_D中定义两个字段:A_ID和A_TYPE,A_TYPE字段表示子类的类型,A_ID参照在子类对应的表中的主键。图14-12显示了表TABLE_D、TABLE_B和TABLE_C的结构。
  

  
图14-12 表TABLE_D、TABLE_B和TABLE_C的结构
相关内容
赞助商链接