Hibernate:重复的键值违反了集合的唯一约束

标签 hibernate jpa

我想导入具有一组专业和一组专业组的配置文件。 Professions 分布在 ProfessionGroups 中。

定义了以下实体:

@Entity
public class Profile extends BaseEntity<Profile> { // B.E. defines id, creation_time,etc..

    @OneToMany(cascade = CascadeType.ALL)
    private Collection<Profession> professions;

    @OneToMany(cascade = CascadeType.ALL)
    private Collection<ProfessionGroup> professionGroups;

    // .. getters and setters
}

@Entity
private class Profession extends BaseEntity<Profession> {

    @Column(unique = true)
    private String name;

    // getters and setters

}

@Entity
public class ProfessionGroup extends BaseEntity<ProfessionGroup> {

    @Column(unique = true)
    private String name;

    @ManyToOne(cascade = CascadeType.All)
    private Collection<Profession> professions;

    // getters and setters
}

以下代码读取一些序列化为 json 的配置文件,并希望将其存储到数据库中:

// ...
Profile p = ...; // read from json using some deserializer
p.getProfessionGroups().forEach(pg -> pg.setProfessions(p.getProfessions());

// ..
ProfileService profileService = ...; // 
profileService.save(profile);

ProfileService 在内部调用 entityManager.persist(...)。 这里的问题是,每当我想将所有职业分配给所有职业组时,我都会收到“重复键值违反唯一约束”。我该怎么做才能安全地存储配置文件,而不会违反唯一键约束。 JPA 显然希望为专业组中的每个条目创建一个新专业。但是,对专业的引用是相同的。调用 merge(...) 没有成功。

最佳答案

问题在于级联的定义,以及 JPA 尤其是 hibernate 如何处理新的实体实例。

调用 entitymanager.persist 时,它会存储和管理实体的状态,但不会存储和管理您传递给 entityManager.persist 的实际对象。

托管实例和传递的参数会不同。因此,如果您手动生成 ID,则使用同一对象调用 entitymanager.persist 两次将导致来自数据库的 DuplicateKeyException,而不是来自 jpa。

要解决这个问题,您需要保留并获取对 Profession 实例的托管实体的引用,您可以在 Profile 和 ProfessionGroup 中使用它们,因此:

Profile profile = loadProfiles();
List<Profession> managedProfessions = profile
             .getProfessions()
             .stream()
             .map((p) -> entityManager.merge(p)) //Note that we use the returned value, since the returned value is what is actually managed, the passed parameter is not, and will be discarded by the persistent-context
             .Collect(Collectors.toList());
profile.setProfessions(managedProfessions);
profile.getProfessionGroups().forEach((gr)->gr.setProfessions(managedProfessions));

profileService.save(profile);

有了这个,您可能想要删除 Cascade.ALL 并只用 Cascade.MERGE 替换它。

关于Hibernate:重复的键值违反了集合的唯一约束,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39067243/

相关文章:

mysql - 在 HQL 查询的 where 子句中进行查询,但不想选择该字段

java - JPA/Hibernate : Sub-typing vs. 策略 'pattern'

java - 使用 Hibernate/JPA 在运行时抛出 AbstractMethodError

java - 分离/逐出具有非持久化子对象的对象树

java - 在 liquibase 上用 JSON 内容重构数据库

java - 是否可以在没有列的情况下在 JPA/Hibernate 中创建和使用序列生成器?

spring - Spring Entity Manager 和 Spring Data Repository 有什么区别?

spring - 如何将 [DDL] 从生成类型标识迁移到 postgres db 的生成类型序列

java - 为什么我的 HQL 查询急于加载惰性关联?

java - 嵌套对象空字段的 Spring 表单绑定(bind)问题