精通Hibernate之映射继承关系七

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

假定与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属性时使用了延迟检索策略:

当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的结构

如果你曾歌颂黎明,那么也请你拥抱黑夜

精通Hibernate之映射继承关系七

相关文章:

你感兴趣的文章:

标签云: