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