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

前言

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

解决

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

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

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

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

List klassList = this.klassRepository.findAll();
List allStudent = this.studentRepository.findAll();
for (Klass klass: klassList) {
    List 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的懒查询机制。
记录一次班级与学生修改为多对多关系的过程_第3张图片
解决办法是在@ManyToMany上关闭懒查询

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

你可能感兴趣的