考虑这种情况.
有地块,有些是住宅用地,有些是商业用地.
还有业主.但是业主只能购买一块地块,它可以是住宅或商业用途.
所以,这是我的代码.
@Entity
@Table(name = "PLOT")
@Inheritance(strategy = InheritanceType.JOINED)
public abstract class Plot {
private int id;
private String number;
private List<Owner> owners = new ArrayList<>();
// getters and setters...
}
@Entity
@Table(name = "RESIDENTIAL_PLOT")
@PrimaryKeyJoinColumn(name = "PLOT_ID")
public class ResidentialPlot extends Plot {
// Some fields
}
@Entity
@Table(name = "COMMERCIAL_PLOT")
@PrimaryKeyJoinColumn(name = "PLOT_ID")
public class CommercialPlot extends Plot {
// Some fields
}
@Entity
@Table(name = "OWNER")
public class Owner {
private int id;
private String name;
private Plot plot;
// getters and setters
}
Run Code Online (Sandbox Code Playgroud)
一切运作良好,但是当我打电话给owner.getPlot()
我期待ResidentialPlot或CommercialPlot实例时,我可以通过使用instanceof运算符来应用适当的操作.但它不满足这两个条件!
我究竟做错了什么?
你没有使用多态,在JPA环境中它会受到更多伤害.
您观察到此行为的原因是因为Hibernate使用了代理.当Hibernate加载一个所有者时,它无法判断该情节是商业用途还是住宅用地.它必须做一个额外的查询才能知道它.因此,它使用惰性代理初始化绘图字段,这是一个动态生成的类的实例,它扩展了Plot,但既不是CommercialPlot也不是ResidentialPlot.
当在代理上调用方法时,代理通过从数据库获取Plot数据来初始化自身,并且它委托给CommercialPlot或ResidentialPlot的实例.
解决方案是使用多态性.将一个抽象方法添加到Plot类(getType(),或者isResidential(),例如),并在两个子类中实现它.如果这是不可能的,因为您需要依赖于实体类型但不应该在实体本身中的业务逻辑,请使用访问者模式.
如果你需要更多细节,我写了一篇关于这个主题的博客文章,但它是用法语写的.也许谷歌翻译可以帮助你.
| 归档时间: |
|
| 查看次数: |
3327 次 |
| 最近记录: |