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

前言 这周做的是alice项目中的一个bug,由于重新分班后,原班级没有学生,对于原班级的某一课程成绩分析也就没有了。
记录一次班级与学生修改为多对多关系的过程
文章图片

比如说我原来是计算机1806班的(计算机专业一共9个班,分别是计算机1801-计算机1809班),由于大专业分流成小专业,我被分为了软件181班,如果班级和学生是一对多的关系,此时计算机1806班已经没有学生了,但是成绩分析是对一个班的某一课程安排的成绩分析,需要获取这个班所有学生这个课程安排的成绩,导致成绩分析不能得到所有学生相关成绩。
解决 我一开始认为改实体间关系比较麻烦,我们要的是班级每个人的成绩数组,由于班级没有了学生,所以每个人的成绩找不到。但是每个人的成绩还在数据库里存着没有丢失,我们换一种方法去查询就好了。这样不用改多对一也可以满足需求。当我快改完时,学长说还想要在班级管理里看到原班级的所有学生,这就不得不去改学生与班级关系多对多了。
此时需要将班级和学生一对多关系变为多对多关系。使得原计算机1806班也可以找到所有分班前所在班内的学生。同时需要再对班级和学生数组设置一对多关系,表示学生当前班级。
记录一次班级与学生修改为多对多关系的过程
文章图片

/** * 学生当前所在班级 */ @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 = "https://www.it610.com/article/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的懒查询机制。
记录一次班级与学生修改为多对多关系的过程
文章图片

解决办法是在@ManyToMany上关闭懒查询
@ManyToMany(fetch = FetchType.EAGER) private List studentList = new ArrayList<>();

    推荐阅读