我开始使用Spring Data JPA在Spring Boot应用程序上设置用户和角色之间的ManyToMany关系。
此关系在User类中定义如下:
@ManyToMany(fetch = FetchType.EAGER, cascade = CascadeType.ALL)
@JoinTable(name="user_role", joinColumns = {@JoinColumn(name="user_id")}, inverseJoinColumns = {@JoinColumn(name="role_id")})
private Set<UserRole> roles;
我使用以下方式创建角色:
@Transactional
private void generateSeedRoles() {
UserRole adminRole = new UserRole(RoleEnum.ADMIN.toString());
userRoleRepository.save(adminRole);
UserRole userRole = new UserRole(RoleEnum.USER.toString());
userRoleRepository.save(userRole);
}
之后为用户分配角色失败:
@Transactional
private void generateSeedUsers() {
UserRole adminRole = userRoleRepository.findUserRoleByRoleName("ADMIN");
User user = User.createUser("user1", "user1@user.com", "pass");
user.setRoles(new HashSet<UserRole>(Arrays.asList(adminRole)));
userRepository.save(user);
}
抛出以下异常(为便于阅读而格式化):
org.springframework.dao.InvalidDataAccessApiUsageException:
detached entity passed to persist: co.feeb.models.UserRole;
nested exception is org.hibernate.PersistentObjectException:
detached entity passed to persist: co.feeb.models.UserRole
但是,如果在创建关系之前保存用户,则可以正常工作:
@Transactional
private void generateSeedUsers() {
UserRole adminRole = userRoleRepository.findUserRoleByRoleName("ADMIN");
User user = User.createUser("user1", "user1@user.com", "pass");
//Save user
userRepository.save(user);
//Build relationship and update user
user.setRoles(new HashSet<UserRole>(Arrays.asList(adminRole)));
userRepository.save(user);
}
两次保存/更新用户对我来说似乎有些不合理。有没有办法将收到的角色分配给新用户而不先保存用户?
答案 0 :(得分:8)
当你保存你的实体时,Spring在内部检查实体是否是新的(我忘记了确切的机制),并发出一个persist(如果是新的)或merge(如果存在的话)。
由于您已经使用Cascade.ALL定义了User与UserRole的关系,因此来自User的任何持久/合并也将级联到UserRole。
考虑到Spring如何实现保存以及上面级联的行为,以下是一些需要考虑的方案:
如果您可以保证您添加到用户关系的任何UserRole始终存在,您只需从级联选项中删除Cascade.Persist即可。否则,我相信在上述两种情况下保存用户时,您必须使用合并 - 通过自定义存储库和entityManager。
关于你对Cascade.ALL的使用,在这种情况下可能没有意义,因为你有@ManyToMany关系,从用户到UserRole级联ALL可能会产生一些不良影响(例如Cascade.Remove会删除每次删除用户时都会收到UserRole。