2

记录一次班级与学生修改为多对多关系的过程

 2 years ago
source link: https://segmentfault.com/a/1190000041217467
Go to the source link to view the article. You can view the picture content, updated content and better typesetting reading experience. If the link is broken, please click the button below to view the snapshot at that time.

记录一次班级与学生修改为多对多关系的过程

image.png
这周做的是alice项目中的一个bug,由于重新分班后,原班级没有学生,对于原班级的某一课程成绩分析也就没有了。
lALPDgfLTvPy0pDNAhzNBG0_1133_540.png
比如说我原来是计算机1806班的(计算机专业一共9个班,分别是计算机1801-计算机1809班),由于大专业分流成小专业,我被分为了软件181班,如果班级和学生是一对多的关系,此时计算机1806班已经没有学生了,但是成绩分析是对一个班的某一课程安排的成绩分析,需要获取这个班所有学生这个课程安排的成绩,导致成绩分析不能得到所有学生相关成绩。

我一开始认为改实体间关系比较麻烦,我们要的是班级每个人的成绩数组,由于班级没有了学生,所以每个人的成绩找不到。但是每个人的成绩还在数据库里存着没有丢失,我们换一种方法去查询就好了。我在score_summary实体中加入对应score数组,这样可以这样不用改多对一也可以满足需求。当我快改完时,学长说还想要在班级管理里看到原班级的所有学生,这就不得不去改学生与班级关系多对多了。
此时需要将班级和学生一对多关系变为多对多关系。使得原计算机1806班也可以找到所有分班前所在班内的学生。同时需要再对班级和学生数组设置一对多关系,表示学生当前班级。
image.png

/**
 * 学生当前所在班级
 */
@ManyToOne
private Klass klass;

/**
 * 学生历史所在班级数组(包含当前所在班级)
 */
@ManyToMany(mappedBy = "studentList")
@JsonView(KlassListJsonView.class)
private List<Klass> klassList = new ArrayList<>();

然后来修改当前变动出现的问题。首先,由于学生对于班级来说由多对一变为多对多,并且由班级一方负责维护,并且在数据库中自动建立了klass_student_list中间表。但是我们从班级中找所有学生没有数据,我们需要先恢复原来班级中所有学生数据。学生中还保留当前班级信息,我们通过这些数据将同一班级中的学生放到班级的学生数组里。这样中间表就有了相关数据。

List<Klass> klassList = this.klassRepository.findAll();
List<Student> allStudent = this.studentRepository.findAll();
for (Klass klass: klassList) {
    List<Student> studentList = new ArrayList<>();
    for (Student student: allStudent) {
        if (student.getKlass() != null && klass.getId().equals(student.getKlass().getId())) {
            studentList.add(student);
        }
    }
    klass.setStudentList(studentList);
}
this.klassRepository.saveAll(klassList);

然后来修改学生更改班级时的代码,为了使业务与技术分离思想,我们采用aop形式修改,也减少了对原代码的改动。思想是在学生变动班级时,在旧班级中将学生删除,在新班级中将学生添加。

@Before(value = "annotationPointCut(id, student)", argNames = "id, student")
public void before(Long id, Student student) {
    Student newStudent = this.studentService.findById(id);
    Klass oldKlass = newStudent.getKlass();
    Klass newKlass = this.klassService.findById(student.getKlass().getId());
    // 如果新班级与旧班级不相等
    if (!oldKlass.getId().equals(newKlass.getId())) {
        // 删除旧班级中的这个学生
        int i =0;
        for (Student student1: oldKlass.getStudentList()) {
            if (student1.getId().equals(student.getId())) {
                oldKlass.getStudentList().remove(i);
                break;
            } else {
                i++;
            }
        }

        // 在新班级中添加这个学生
        newKlass.getStudentList().add(newStudent);
        // 保存
        this.klassService.save(oldKlass);
        this.klassService.save(newKlass);
    }
}

但是重新分班这种情况我不知道是怎么操作的,我还需要跟学长沟通,我也需要用aop修改这种情况,只需设置添加新班级所有学生就可以了。

可能遇到的问题

我在初始化班级内所有学生数据的时候遇到了一个问题,更新完后中间表没有任何数据,然后我打断点看问题,发现在多对多查询班级时并不会将班级里所有学生查出来,这是hibernate的懒查询机制。
lALPDgtYzZlnv2fNAZPNBc8_1487_403.png
解决办法是在@ManyToMany上关闭懒查询

@ManyToMany(fetch = FetchType.EAGER)
private List<Student> studentList = new ArrayList<>();

About Joyk


Aggregate valuable and interesting links.
Joyk means Joy of geeK