Hibernate继承 - 获取超类实例并转换为子类

Aks*_*hat 7 java hibernate

考虑这种情况.

有地块,有些是住宅用地,有些是商业用地.

还有业主.但是业主只能购买一块地块,它可以是住宅或商业用途.

所以,这是我的代码.

@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() 我期待ResidentialPlotCommercialPlot实例时,我可以通过使用instanceof运算符来应用适当的操作.但它不满足这两个条件!

我究竟做错了什么?

JB *_*zet 7

你没有使用多态,在JPA环境中它会受到更多伤害.

您观察到此行为的原因是因为Hibernate使用了代理.当Hibernate加载一个所有者时,它无法判断该情节是商业用途还是住宅用地.它必须做一个额外的查询才能知道它.因此,它使用惰性代理初始化绘图字段,这是一个动态生成的类的实例,它扩展了Plot,但既不是CommercialPlot也不是ResidentialPlot.

当在代理上调用方法时,代理通过从数据库获取Plot数据来初始化自身,并且它委托给CommercialPlot或ResidentialPlot的实例.

解决方案是使用多态性.将一个抽象方法添加到Plot类(getType(),或者isResidential(),例如),并在两个子类中实现它.如果这是不可能的,因为您需要依赖于实体类型但不应该在实体本身中的业务逻辑,请使用访问者模式.

如果你需要更多细节,我写了一篇关于这个主题的博客文章,但它是用法语写的.也许谷歌翻译可以帮助你.