前言
这周做的是alice项目中的一个bug,由于重新分班后,原班级没有学生,对于原班级的某一课程成绩分析也就没有了。
比如说我原来是计算机1806班的(计算机专业一共9个班,分别是计算机1801-计算机1809班),由于大专业分流成小专业,我被分为了软件181班,如果班级和学生是一对多的关系,此时计算机1806班已经没有学生了,但是成绩分析是对一个班的某一课程安排的成绩分析,需要获取这个班所有学生这个课程安排的成绩,导致成绩分析不能得到所有学生相关成绩。
解决
我一开始认为改实体间关系比较麻烦,我们要的是班级每个人的成绩数组,由于班级没有了学生,所以每个人的成绩找不到。但是每个人的成绩还在数据库里存着没有丢失,我们换一种方法去查询就好了。我在score_summary实体中加入对应score数组,这样可以这样不用改多对一也可以满足需求。当我快改完时,学长说还想要在班级管理里看到原班级的所有学生,这就不得不去改学生与班级关系多对多了。
此时需要将班级和学生一对多关系变为多对多关系。使得原计算机1806班也可以找到所有分班前所在班内的学生。同时需要再对班级和学生数组设置一对多关系,表示学生当前班级。
/**
* 学生当前所在班级
*/
@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的懒查询机制。
解决办法是在@ManyToMany上关闭懒查询
@ManyToMany(fetch = FetchType.EAGER)
private List<Student> studentList = new ArrayList<>();
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。