JPA实现多对多关系

简介: JPA实现多对多关系

概念说明

 多对多关系是指两个实体之间存在多对多的关联关系。在数据库中,多对多关系无法直接表示,需要通过中间表来实现。

 举个例子,假设有两个实体类:学生(Student)和课程(Course)。一个学生可以选择多门课程,而一门课程也可以被多个学生选择。这就是一个多对多的关系。

 在数据库中,可以创建三张表来表示多对多关系:学生表(students)、课程表(courses)和中间表(student_course)。中间表中存储了学生和课程之间的关联关系,它包含了学生的ID和课程的ID。

 通过中间表,可以将多对多关系拆分为两个一对多关系。学生和课程之间的关系可以被拆分为学生和中间表之间的一对多关系,以及课程和中间表之间的一对多关系。

优势利弊

 多对多关系的优点是能够灵活地表示实体之间的复杂关联关系。例如,在上述的学生和课程的例子中,一个学生可以选择多门课程,而一门课程也可以被多个学生选择。这种关系在实际应用中非常常见,通过多对多关系可以方便地表示和操作这种复杂的关联关系。

 多对多关系也有一些限制和注意事项。例如,中间表的设计和维护需要一定的注意,以确保关联关系的正确性和一致性。此外,多对多关系可能会导致性能问题,因为查询和操作涉及多个表的关联。因此,在设计和使用多对多关系时,需要仔细考虑和评估实际需求和性能要求。

实现方式

通过两个@ManyToMany注解实现

类图

c8d0a4eb6c7d4b7f90578bbeccc6c99f.png

代码

学生类

@Entity
public class Student {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
    private String name;
    @ManyToMany
    @JoinTable(name = "student_course",
               joinColumns = @JoinColumn(name = "student_id"),
               inverseJoinColumns = @JoinColumn(name = "course_id"))
    private List<Course> courses;
}

在Student实体类中,使用@ManyToMany注解来表示多对多关系,并通过@JoinTable注解指定中间表的名称和关联字段。joinColumns指定了中间表中与Student实体关联的外键字段,inverseJoinColumns指定了中间表中与Course实体关联的外键字段。

课程类

@Entity
public class Course {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
    private String name;
    @ManyToMany(mappedBy = "courses")
    private List<Student> students;
}

 在Course实体类中,使用@ManyToMany(mappedBy = “courses”)注解来表示多对多关系,并通过mappedBy属性指定了关联的属性名,即Student实体类中的courses属性。

中间表

27ba69a03d1c42dabc9fc6735f2a8cac.png

12bfbc55861b431da98c6e6416f391f3.png

 通过使用@ManyToMany的方式我们确实可以实现多对多的效果,但是我们没有办法在中间表中添加其他字段,中间表中只有学生的id和课程id。如果我们想要添加一些比如创建时间、更新时间、是否删除的字段是没有办法添加的。如果我们没有需要添加字段的需求那我们直接通过@ManyToMany 的方式即可。如果我们需要添加一些其他的字段就需要调整我们的实现方式。下面我们通过两个@OneToMany和两个@ManyToOne注解来解决这个问题。

通过@OneToMany和@ManyToOne注解实现

类图

10b092531d7743f686fde7ce39bcc8cf.png

代码

学生类

@Entity
public class Student {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
    private String name;
    @OneToMany(mappedBy = "student")
    private List<StudentCourse> studentCourses;

课程类

@Entity
public class Course {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
    private String name;
    @OneToMany(mappedBy = "course")
    private List<StudentCourse> studentCourses;
}

学生和课程关系类

@Entity
@Table(name = "student_course")
public class StudentCourse {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
    @ManyToOne
    @JoinColumn(name = "student_id")
    private Student student;
    @ManyToOne
    @JoinColumn(name = "course_id")
    private Course course;
    private int grade;
}

这样,通过以上的配置,JPA会自动创建中间表,并维护学生和课程之间的关联关系,同时在中间表中添加了grade字段。你可以通过访问StudentCourse实体类来操作中间表的数据,包括添加学生选课、查询学生所选的课程以及对应的成绩等。

少走弯路

 当实现了多对多之后我们会发现在具体应用这些对象的时候会出现循环引用的问题,可能会导致堆栈溢出的问题。参考下面博客来解决循环引用带来的问题:

https://blog.csdn.net/weixin_45490198/article/details/131795009

总结提升

在两个实体类中使用@ManyToMany注解来表示多对多关系,并通过@JoinTable注解指定中间表的名称和关联字段。

JPA会自动创建中间表,并维护两个实体类之间的关联关系。

如果需要在中间表中添加其他字段,可以创建一个新的实体类来表示中间表,并在两个实体类中使用@OneToMany和@ManyToOne注解来表示与中间表的一对多关系。


相关文章
|
5月前
|
JSON Java API
Java一分钟之-JPA实体关系:一对一, 一对多, 多对多
【6月更文挑战第14天】Java Persistence API (JPA) 的 ORM 规范简化了数据库操作,重点是实体关系映射。本文讨论了三种主要关系:一对一、一对多和多对多。对于每种关系,都指出了常见问题(如循环引用、懒加载异常)和避免策略(使用注解解决循环引用,明确级联操作边界等)。同时,提供了示例代码以展示如何在实践中设置这些关系。理解并妥善处理这些问题能提升开发效率和数据准确性。
228 1
|
数据可视化 uml
UML图讲解(关联关系,单向关联,双向关联,自关联,组合关系,依赖关系,继承关系,实现关系)
UML图讲解,关联关系,单向关联,双向关联,自关联,组合关系,依赖关系,继承关系,实现关系。
3358 0
UML图讲解(关联关系,单向关联,双向关联,自关联,组合关系,依赖关系,继承关系,实现关系)
|
容器
getChildFragmentManager和getsupportFragmentManager和getFragmentManager的关系
getChildFragmentManager和getsupportFragmentManager和getFragmentManager的关系
|
存储 关系型数据库 数据库
[译] 如何用 Room 处理一对一,一对多,多对多关系?
[译] 如何用 Room 处理一对一,一对多,多对多关系?
[译] 如何用 Room 处理一对一,一对多,多对多关系?
|
开发者
多对多关系 | 学习笔记
快速学习多对多关系。
149 0
多对多关系 | 学习笔记
举一个多对多关联的例子,并说明如何实现多对多关联映射
例如:商品和订单、学生和课程都是典型的多对多关系。可以在实体类上通过@ManyToMany注解配置多对多关联或者通过映射文件中的和标签配置多对多关联,但是实际项目开发中,很多时候都是将多对多关联映射转换成两个多对一关联映射来实现的。
1515 0