27

データベースに 3 つのテーブルがあります:StudentsCoursesStudents_Courses

学生は複数のコースを持つことができ、コースは複数の学生を持つことができます。と の間には多対多の関係がStudentsありCoursesます。

プロジェクトに 3 つのケースがあり、テーブルにコースが追加されていCoursesます。

  • (a) ユーザーを追加すると、問題なく保存されます。
  • (b) 学生のコースを追加するとUser_Courses、期待される動作に新しい行が作成されます。
  • (c) 生徒を削除しようとすると、 と の適切なレコードが削除されますStudentsが、不要なレコードStudents_Coursesも削除されます。Coursesコースにユーザーがいない場合でも、コースがそこにあることを望みます。

以下は、テーブルと注釈クラスのコードです。

    CREATE TABLE `Students` (
    `StudentID` INT(11) NOT NULL AUTO_INCREMENT,
    `StudentName` VARCHAR(50) NOT NULL 
    PRIMARY KEY (`StudentID`)
)

CREATE TABLE `Courses` (
    `CourseID` INT(11) NOT NULL AUTO_INCREMENT,
    `CourseName` VARCHAR(50) NOT NULL 
    PRIMARY KEY (`CourseID`)
)

CREATE TABLE `Student_Courses` (
    `StudentId` INT(10) NOT NULL DEFAULT '0',
    `CourseID` INT(10) NOT NULL DEFAULT '0',
    PRIMARY KEY (`StudentId`, `CourseID`),
    INDEX `FK__courses` (`CourseID`),
    INDEX `StudentId` (`StudentId`),
    CONSTRAINT `FK__courses` FOREIGN KEY (`CourseID`) REFERENCES `courses` (`CourseID`) ON DELETE NO ACTION,
    CONSTRAINT `FK_students` FOREIGN KEY (`StudentId`) REFERENCES `students` (`StudentId`)
)

これは、Hibernate によって生成された Java コードです。

@Entity
@Table(name = "Students")
public class Students implements java.io.Serializable {

    private Integer StudentID;
     private String Students;
    private Set<Courses> Courseses = new HashSet<Courses>(0);

    public Students() {
    }


    @Id
    @GeneratedValue(strategy = IDENTITY)
    @Column(name = "StudentID", unique = true, nullable = false)
    public Integer getStudentID() {
        return this.StudentID;
    }

    public void setStudentID(Integer StudentID) {
        this.StudentID = StudentID;
    }

    @Column(name = "Students", nullable = false, length = 50)
    public String getCampaign() {
        return this.Students;
    }

    public void setCampaign(String Students) {
        this.Students = Students;
    }


 @ManyToMany(cascade = {CascadeType.ALL}, fetch = FetchType.LAZY)
    @JoinTable(name = "Student_Courses", joinColumns = {
        @JoinColumn(name = "StudentId", nullable = false, updatable = false)}, inverseJoinColumns = {
        @JoinColumn(name = "CourseID", nullable = false, updatable = false)})
    public Set<Courses> getCourseses() {
        return this.Courseses;
    }

     public void setCourseses(Set<Courses> Courseses) {
        this.Courseses = Courseses;
    }

    }


    @Entity
@Table(name = "Courses")
public class Courses implements java.io.Serializable {

  private Integer CourseID;
    private String CourseName;
     private Set<Students> Studentses = new HashSet<Students>(0);

    public Courses() {
    }

    @Id
    @GeneratedValue(strategy = IDENTITY)
    @Column(name = "CourseID", unique = true, nullable = false)
    public Integer getCourseID() {
        return this.CourseID;
    }

    public void setCourseID(Integer CourseID) {
        this.CourseID = CourseID;
    }

     @Column(name = "CourseName", nullable = false, length = 100)
    public String getCourseName() {
        return this.CourseName;
    }

    public void setCourseName(String CourseName) {
        this.CourseName = CourseName;
    }

    @ManyToMany(cascade = CascadeType.ALL, fetch = FetchType.LAZY, mappedBy = "Courseses")
    public Set<Students> getStudentses() {
        return this.Studentses;
    }

    public void setStudentses(Set<Students> Studentses) {
        this.Studentses = Studentses;
    }

    }

私が説明したことをどのように達成できますか?ウェブ上で適切なドキュメントを見つけることができませんでした。

4

3 に答える 3

17

あなたが私に言ったことに基づいて、Student の getCourseses メソッドで cascade=CascadeType.ALL を使用したくありません。Hibernate カスケードはデータベース カスケードと同じではないことに注意してください。カスケードがない場合でも、Hibernate は Students_Courses レコードを削除します。

Hibernate カスケードを考える最良の方法は、エンティティで操作を呼び出し、その操作がカスケード リストにリストされている場合、その操作はすべての子エンティティで呼び出されるということです。

たとえば、Student に対して delete を呼び出すと、delete は Courses のカスケード リストにあるため、Hibernate はその Student が参照する各 Course エンティティに対して delete を呼び出します。そのため、コース レコードが消えています。

データベースのカスケードについて心配する必要はありません。Hibernate が独自に処理します。

于 2013-01-30T13:26:40.313 に答える