Hibernate Lazy加载导致的OO不完全实现

转自: http://www.sulong.info/archives/335

 

对象和关系型数据库之间存在着根本不同, ORM(对象/关系映射)试图把对象映射到关系型数据库的表结构上,从而简化存储对象到数据库及从数据库中恢复对象的复杂性,让操作数据库就像操作对象一样简单。Hibernate估计是众多ORM框架中最成熟的了,但是在处理类的继承关系时还是会有一些问题。

延迟加载与继承
先看如下的代码, 演示这一映射关系:

@Entity
@Table(name = "a")
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(
		name = "disc", 
		columnDefinition = "varchar(4)", 
		discriminatorType = DiscriminatorType.STRING)
public abstract class A {
 @Id
 private int oid;
 private String name;
}
 
@Entity
@DiscriminatorValue("b")
public class B extends A {
private String age;
}
 
@Entity
@DiscriminatorValue("c")
public class C extends B {
private String sex;
}
 
@Entity
@Table(name = "d")
public class D {
  @Id
  private int oid;
  @ManyToOne(fetch = FetchType.LAZY)
  private A a;
  public A getA() {return a}
}
 

 

当查询 D类时,应为 D.a 是延迟加载的,所以没有查询过 a表,自然就不知道该记录对应的disc为什么值,也不知道A具体是哪种子类类型。但是查询出的d对象,如果 d.a != null d.a必须有值。Hibernate会生成一个A的子类,并用它的实例作为 d.a 当你真的要访问 d.a 的方法或域时,hibernate才会真正的去查询 a 表。 这其实就是hibernate实现延迟加载的原理。但在本例中,如果你执行如下代码就会遇到 ClassCastException,

D d = session.load(D.class, 1);
C c = (C) d.getA(); //即使d.getA()确实是C类型,也会ClassCastException 
 

hibernate生成的是A的子类,而不是B或C的子类,所以在类型转换的时候出错了。要解决这一问题,要么不要用延迟加载, 但是性能上面很可能出问题;要么不要用继承,向下面这样:

@Entity
@Table(name = "a")
public abstract class A {
 @Id
 private int oid;
 
 private String name;
 private String age;
 private String sex;
}
 
 
@Entity
@Table(name = "d")
public class D {
  @Id
  private int oid;
  @ManyToOne(fetch = FetchType.LAZY)
  private A a;
  public A getA() {return a}
} 
 

这样实际上是限制OO的能力;要么每次都显示的重新查询改对象,向这样:

D d = session.load(D.class, 1);
C c = session.load(C.class, d.getA().getOid()) //重新查,指明类型 
 

这样做既不方便,又不OO。

看来要想让hibernate的映射类保持对象的特性,还是有些困难的。很多时候,这些映射类最终就变成了只负责映射功能的数据库表对象了

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值