如何在Hibernate中从非实体子类中持久化实体

时间:2011-09-19 07:34:44

标签: java hibernate jpa orm annotations

我正在尝试将实体扩展为非实体,用于填充超类的字段。问题是,当我尝试保存它时,Hibernate会抛出MappingException。这是因为即使我将ReportParser转换为Report,运行时实例仍然是ReportParser,因此Hibernate抱怨它是一个未知的实体。

@Entity
@Table(name = "TB_Reports")
public class Report
{
   Long id;
   String name;
   String value;

   @Id
   @GeneratedValue
   @Column(name = "cReportID")
   public Long getId()
   {
      return this.id;
   }

   public void setId(Long id)
   {
      this.id = id;
   }

   @Column(name = "cCompanyName")
   public String getname()
   {
      return this.name;
   }

   public void setName(String name)
   {
      this.name = name;
   }

   @Column(name = "cCompanyValue")
   public String getValue()
   {
      return this.name;
   }

   public void setValue(String value)
   {
      this.value = value;
   }
}

ReportParser仅用于填写字段。

public class ReportParser extends report
{
   public void setName(String htmlstring)
   {
      ...
   }

   public void setValue(String htmlstring)
   {
      ...
   }
}

尝试将其投射到报告并保存

...
ReportParser rp = new ReportParser();
rp.setName(unparsed_string);
rp.setValue(unparsed_string);
Report r = (Report)rp;
this.dao.saveReport(r);

在我转移到ORM之前,我已经使用过这个模式,但是我无法弄清楚如何使用Hibernate。有可能吗?

2 个答案:

答案 0 :(得分:3)

是否绝对有必要对实体进行子类化?您可以使用构建器模式:

public class ReportBuilder {
    private Report report;
    public ReportBuilder() {
        this.report = new Report();
    }
    public ReportBuilder setName(String unparsedString) {
        // do the parsing
        report.setName(parsedString);
        return this;
    }
    public ReportBuilder setValue(String unparsedString) {
        // do the parsing
        report.setValue(parsedString);
        return this;
    }
    public Report build() {
        return report;
    }
}

Report report = new ReportBuilder()
                   .setName(unparsedString)
                   .setValue(unparsedString)
                   .build();
dao.saveReport(report);

答案 1 :(得分:0)

除非要制作更专业的实体类,否则不应该扩展实体类。与实体相关的注释保留在子类中,因此Hibernate会感到困惑。

将业务逻辑放在实体类中也是值得商榷的 - 你必须要知道JPA实现者,比如Hibernate,可能(并且通常会)通过生成的代理来运行你的getter / setter。内部复杂的逻辑可能会遇到难以追查的问题。