|
|
|
| 您的位置:首页>>Hibernate>>精通Hibernate之映射继承关系七(图) |
|
|
精通Hibernate之映射继承关系七(图)
|
| 2006-04-05 来源:赛迪网 作者:孙卫琴 |
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的结构 |
|
|
|