JPA 持久化与现有实体有关系的新实体



>我有两个具有@ManyToMany关系的实体,让我们举一个常见的例子学生和课程,我正在保留一门带有学生列表的新课程,如果列表中有新学生,它应该级联新学生,如果它存在,它应该更新它。

对于新学生和新课程的情况,它可以正常工作,在新课程和现有学生的情况下,但是我从数据库中抛出了一个唯一的约束冲突。关系类似于以下内容。

@Entity
@Table
public class Course implements Serializable {
    @Id
    @Column(name = "CRS_ID")
    @SequenceGenerator(name = "rcrsSeq", sequenceName = "CRS_SEQ", allocationSize = 1)
    @GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "rcrsSeq")
    private Long id;
    @ManyToMany(cascade = {
        CascadeType.PERSIST,
        CascadeType.MERGE,
        CascadeType.REFRESH,
        CascadeType.DETACH
    })
    @JoinTable(name = "student_courses",
        uniqueConstraints = @UniqueConstraint(columnNames = {"STU_ID", "CRS_ID"}),
        joinColumns = {@JoinColumn(name = "CRS_ID")},
        inverseJoinColumns = {@JoinColumn(name = "STU_ID")}
    )
    private List<Student> students;
    //getters and setters
}

@Entity
@Table(uniqueConstaints = @UniqueConstraint(columnNames = {"username"}))
public class Student implements Serializable {
    @Id
    @Column(name = "STU_ID")
    @SequenceGenerator(name = "rstuSeq", sequenceName = "STU_SEQ", allocationSize = 1)
    @GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "rstuSeq")
    private Long id;
    @ManyToMany(mappedBy = "students")
    private List<Course> courses;

    private String userName;
    //getters and setters
    // equals + hashcode based on userName
}

在坚持之前,我尝试使用实体管理器查找学生,如果该学生存在,请使用该实例而不是新的学生实例。这也不起作用,它仍然尝试插入而不是更新

你写的

但是,在新

课程和现有学生的情况下,我从数据库中抛出了一个独特的约束冲突。

如果要保留新的Course对象和现有的Student对象,则应执行以下操作:

  1. 创建Course的实例:

    Course course = new Course(...);
    
  2. 获取现有Student

    Student student = entityManager.find(Student.class, id_of_student);
    
  3. 按如下方式连接关系:

    course.getStudents.add(student);
    student.getCourse().add(course);
    
  4. 实际上,您应该调用entityManager.persist()来保存新实体。但在这种情况下,这将不起作用,因为student不是新的实体实例,并且persist()不会保存现有的实体实例。因此,即使语义不正确,您也可以使用entityManager.merge()(因为merge应该用于分离的实体):

    entityManager.merge(course);
    

为了使此功能正常工作,您应该调用事务中的所有实体管理器方法。

最新更新