从Hibernate中获取没有主键的表中的实体

时间:2018-01-08 13:46:12

标签: java mysql hibernate

我目前正在开展一个项目,我试图从没有主键的表中获取一系列enities(dk_systemtherapie_merkmale)。该表是1:n与另一个表(dk_systemtherapie)相关。请参阅表结构的屏幕截图。 获取dk_systemtherapie的条目时,程序将获取Collection" dkSystemtherapieMerkmalesById"。但是,第一个表条目的获取频率与表中实际条目的数量一样多。它永远不会从dk_systemtherapie_merkmale中获取其他条目。我认为它与hibernate在条目之间不同的事实有关,但我不知道如何修复它。

Table schema

我创建了两个相应的实体类,dk_systemtherapie:

@Entity
@Table(name = "dk_systemtherapie", schema = "***", catalog = "")
public class DkSystemtherapieEntity {
    private int id;
    private Collection<DkSystemtherapieMerkmaleEntity> dkSystemtherapieMerkmalesById;

    @Id
    @Column(name = "id")
    public int getId() {
        return id;
    }

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

 @OneToMany(mappedBy = "dkSystemtherapieByEintragId")
    public Collection<DkSystemtherapieMerkmaleEntity> getDkSystemtherapieMerkmalesById() {
        return dkSystemtherapieMerkmalesById;
    }

    public void setDkSystemtherapieMerkmalesById(Collection<DkSystemtherapieMerkmaleEntity> dkSystemtherapieMerkmalesById) {
        this.dkSystemtherapieMerkmalesById = dkSystemtherapieMerkmalesById;
    }
}

这是第二个,没有主键访问表,dk_systhemtherapie_merkmale:

@Entity @IdClass(DkSystemtherapieMerkmaleEntity.class)
@Table(name = "dk_systemtherapie_merkmale", schema = "***", catalog = "")
public class DkSystemtherapieMerkmaleEntity implements Serializable {
    @Id private Integer eintragId;
    @Id private String feldname;
    @Id private String feldwert;
    private DkSystemtherapieEntity dkSystemtherapieByEintragId;

    @Basic
    @Column(name = "eintrag_id")
    public Integer getEintragId() {
        return eintragId;
    }

    public void setEintragId(Integer eintragId) {
        this.eintragId = eintragId;
    }

    @Basic
    @Column(name = "feldname")
    public String getFeldname() {
        return feldname;
    }

    public void setFeldname(String feldname) {
        this.feldname = feldname;
    }

    @Basic
    @Column(name = "feldwert")
    public String getFeldwert() {
        return feldwert;
    }

    public void setFeldwert(String feldwert) {
        this.feldwert = feldwert;
    }

    @Id
    @ManyToOne
    @JoinColumn(name = "eintrag_id", referencedColumnName = "id")
    public DkSystemtherapieEntity getDkSystemtherapieByEintragId() {
        return dkSystemtherapieByEintragId;
    }

    public void setDkSystemtherapieByEintragId(DkSystemtherapieEntity dkSystemtherapieByEintragId) {
        this.dkSystemtherapieByEintragId = dkSystemtherapieByEintragId;
    }
 }

1 个答案:

答案 0 :(得分:0)

我认为问题与Hibernate使用以下注释作为从数据库中获取数据的唯一ID一样。

@Id
@ManyToOne
@JoinColumn(name = "eintrag_id", referencedColumnName = "id")
public DkSystemtherapieEntity getDkSystemtherapieByEintragId() {
    return dkSystemtherapieByEintragId;
}

这导致了这样的问题:当获得多个具有相同id的条目(因为id不是唯一的)时,您将获得您想要的条目数,但是hibernate总是获取此id的第一个条目。所以实际上你正在获得一些参赛作品。

那么如何解决这个问题呢?

根据这个问题:Hibernate and no PK,有两种解决方法实际上只有在您的表中没有NULL条目时才会起作用(否则返回的对象也将为NULL)并且没有1 :n关系。据我了解,hibernate不支持没有主键(documentation)的表上的实体。为了确保获得正确的结果,我建议使用NativeQuery。

从DkSystemtherapieMerkmaleEntity.java中删除注释和private DkSystemtherapieEntity dkSystemtherapieByEintragId;(包括bean)并添加构造函数。

public class DkSystemtherapieMerkmaleEntity {
    private Integer eintragId;
    private String feldname;
    private String feldwert;

    public DkSystemtherapieMerkmaleEntity(Integer eintragId, String feldname, String feldwert) {
        this.eintragId = eintragId;
        this.feldname = feldname;
        this.feldwert = feldwert;
    }

    public Integer getEintragId() {
        return eintragId;
    }

    public void setEintragId(Integer eintragId) {
        this.eintragId = eintragId;
    }

    public String getFeldname() {
        return feldname;
    }

    public void setFeldname(String feldname) {
        this.feldname = feldname;
    }

    public String getFeldwert() {
        return feldwert;
    }

    public void setFeldwert(String feldwert) {
        this.feldwert = feldwert;
    }
}

从DkSystemtherapieEntity.java中删除private Collection<DkSystemtherapieMerkmaleEntity> dkSystemtherapieMerkmalesById;(包括bean)。

当您需要获取特定eintrag_id的条目时,请使用以下方法代替DkSystemtherapieEntity.java中的Collection。

public List<DkSystemtherapieMerkmaleEntity> getDkSystemtherapieMerkmaleEntities(int id) {
    Transaction tx = session.beginTransaction();
    String sql = "SELECT * FROM dk_systemtherapie_merkmale WHERE eintrag_id =:id";

    List<Object[]> resultList;
    resultList = session.createNativeQuery(sql)
            .addScalar("eintrag_id", IntegerType.INSTANCE)
            .addScalar("feldname", StringType.INSTANCE)
            .addScalar("feldwert", StringType.INSTANCE)
            .setParameter("id", id).getResultList();

    tx.commit();

    List<DkSystemtherapieMerkmaleEntity> merkmale = new ArrayList<>();
    for (Object[] o : resultList) {
        merkmale.add(new DkSystemtherapieMerkmaleEntity((Integer) o[0], (String) o[1], (String) o[2]));
    }

    return merkmale;
}

致电getDkSystemtherapieMerkmaleEntities(dkSystemtherapieEntityObject.getid())而非getDkSystemtherapieMerkmalesById()