gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-12-02 17:25:15 26 4
gpt4 key购买 nike

我想导入具有一组专业和一组专业组的配置文件。 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/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com