Company与Employee类之间为一对多多态关联关系如果继承关系树的根类对应一个表或者每个类对应一个表那么就能映射Company类的employees集合本节介绍如何映射多对一多态关联如图所示ClassD与ClassA为多对一多态关联关系
图 ClassD与ClassA为多对一多态关联关系ClassAClassB和ClassC构成了一棵继承关系树如果继承关系树的根类对应一个表或者每个类对应一个表那么可以按以下方式映射ClassD的a属性
<manytoone name=a
class=ClassA
column=A_ID
cascade=saveupdate />
假定与ClassD对应的表为TABLE_D与ClassA对应的表为TABLE_A在TABLE_D中定义了外键A_ID它参照TABLE_A表的主键
ClassD对象的a属性既可以引用ClassB对象也可以引用ClassC对象例如
tx = sessionbeginTransaction();
ClassD d=(ClassD)sessionget(ClassDid);
ClassA a=dgetA();
if(a instanceof ClassB)
Systemoutprintln(((ClassB)a)getB());
if(a instanceof ClassC)
Systemoutprintln(((ClassC)a)getC());
mit();
以下代码在映射ClassD类的a属性时使用了延迟检索策略
<manytoone name=a
class=ClassA
column=A_ID
lazy=true
cascade=saveupdate />
当Hibernate加载ClassD对象时它的属性a引用ClassA的代理类实例在这种情况下如果对ClassA的代理类实例进行类型转换会抛出ClassCastException
ClassA a=dgetA();
ClassB b=(ClassB)a; //抛出ClassCastException
解决以上问题的一种办法是使用Sessionload()方法
ClassA a=dgetA();
ClassB b=(ClassB)sessionload(ClassBclassagetId());
Systemoutprintln(bgetB());
当执行Session的load()方法时Hibernate并不会访问数据库而是仅仅返回ClassB的代理类实例这种解决办法的前提条件是必须事先知道ClassD对象实际上和ClassA的哪个子类的对象关联
解决以上问题的另一种办法是显式使用迫切左外连接检索策略避免Hibernate创建ClassA的代理类实例而是直接创建ClassA的子类的实例
tx = sessionbeginTransaction();
ClassD d=(ClassD)sessioncreateCriteria(ClassDclass)
add(Expressioneq(idid))
setFetchMode(aFetchModeEAGER)
uniqueResult();
ClassA a=dgetA();
if(a instanceof ClassB)
Systemoutprintln(((ClassB)a)getB());
if(a instanceof ClassC)
Systemoutprintln(((ClassC)a)getC());
mit();
如果继承关系树的具体类对应一个表为了表达ClassD与ClassA的多态关联需要在TABLE_D中定义两个字段A_ID和A_TYPEA_TYPE字段表示子类的类型A_ID参照在子类对应的表中的主键图显示了表TABLE_DTABLE_B和TABLE_C的结构
educitycn/img_///jpg>图 表TABLE_DTABLE_B和TABLE_C的结构