17370845950

Hibernate 嵌入类中覆盖 "mappedBy" 属性

本文将深入探讨 Hibernate 中在嵌入类中覆盖 "mappedBy" 属性的问题。通过示例代码展示了尝试在 @Embeddable 类中使用 @OneToMany 关系并指定 mappedBy 属性的场景。根据 JPA 规范,嵌入类中的 @OneToMany 关系不能使用 mappedBy 属性,因为嵌入类必须是关系的拥有方,并且关系必须通过外键映射。本文详细解释了这一限制的原因,并提供了替代方案的思路。

嵌入类与关系映射

在 Hibernate 中,@Embeddable 注解用于将一个类标记为可嵌入到其他实体类中。这允许将一些通用的属性组合成一个可重用的单元。然而,在使用嵌入类时,需要注意 JPA 规范对关系映射的限制。

考虑以下示例:

@Entity
public class Parent1 {

    @Embedded
    private Common common;
}

@Entity
public class Parent2 {

    @Embedded
    private Common common;
}

@Entity
public class OtherEntity {

    @ManyToOne
    @JoinColumn(name="p_id_1")
    private Parent1 parent1;

    @ManyToOne
    @JoinColumn(name="p_id_2")
    private Parent2 parent2;
}

@Embeddable
public class Common {

    @OneToMany(mappedBy="IT_DEPENDS") //??? 尝试使用 mappedBy
    private OtherEntity other;

}

以上代码尝试在 Common 嵌入类中使用 @OneToMany 关系,并尝试通过 mappedBy 属性指定关系的映射方。

JPA 规范的限制

根据 JPA 规范的 2.7 章节,嵌入类(包括嵌套的嵌入类)如果包含在元素集合中,则不能包含元素集合,也不能包含指向实体的关系,除非是多对一或一对一关系。此外,嵌入类必须是关系的拥有方,并且关系必须通过外键映射。

这意味着在 @Embeddable 类中,不能使用 @OneToMany 关系,并且不能指定 mappedBy 属性。这是因为嵌入类必须是关系的拥有方,而不是被拥有方。

替代方案

由于 JPA 规范的限制,直接在嵌入类中使用 mappedBy 属性是不可能的。但是,可以考虑以下替代方案:

  1. 将关系移动到实体类中: 将 @OneToMany 关系从 Common 嵌入类移动到 Parent1 和 Parent2 实体类中。这样可以避免嵌入类的限制。

    @Entity
    public class Parent1 {
    
        @Id
        private Long id;
    
        @OneToMany(mappedBy = "parent1")
        private List otherEntities;
    
        @Embedded
        private Common common;
    }
    
    @Entity
    public class OtherEntity {
    
        @Id
        private Long id;
    
        @ManyToOne
        @JoinColumn(name = "parent1_id")
        private Parent1 parent1;
    }
  2. 使用双向 @ManyToOne 关系: 如果需要从 OtherEntity 访问 Parent1 或 Parent2,可以使用双向 @ManyToOne 关系。在这种情况下,OtherEntity 将拥有指向 Parent1 或 Parent2 的外键。

    @Entity
    public class Parent1 {
    
        @Id
        private Long id;
    
        @Embedded
        private Common common;
    }
    
    @Entity
    public class OtherEntity {
    
        @Id
        private Long id;
    
        @ManyToOne
        @JoinColumn(name = "parent1_id")
        private Parent1 parent1;
    }

总结

在 Hibernate 中,虽然嵌入类提供了代码重用的便利性,但在关系映射方面存在一定的限制。尤其是在使用 @OneToMany 关系时,不能在嵌入类中使用 mappedBy 属性。因此,需要根据具体的需求选择合适的替代方案,例如将关系移动到实体类中,或者使用双向 @ManyToOne 关系。理解这些限制和替代方案对于正确地使用 Hibernate 进行对象关系映射至关重要。