数据库查询一直是提高程序速度的瓶颈之一,本人也遇到了因为数据库数据查询过久导致影响失败,在这里记录以下解决的方法。
描述
该功能为统计功能,根据参数查找出符合的器具信息,然后再根据统计信息得出统计数据。就好比要统计某个年级学生的优秀,良好,及格率。看起来简单容易实现,但是到了测试环境中,却出现了错误:查询过久得不到响应。
排查
经过打断点查看,发现系统问题出在了查询数据的过程中。经过排查是数据过大,并且要连接的表过多,导致查询时间花费过大,前台得不到响应。
hibernate
在查询时,总会自动把所有关联的数据查询出来,而关联的数据又有关联的实体,这样一样导致要连接查询的表过多。就好比学生,教师,学院这三个实体:
class Student {
......
@ManyToOne
Teacher teacher; // 所属教师
@OneToOne
Score score; // 所属成绩
}
class Teacher {
......
@ManToOne
College college; // 所属学院
}
class College {
.......
}
当统计学生成绩时,我们并不需要教师的信息,但是hibernate
在查询时会自动来连接teacher表把教师的信息也查询出来,同时查询教师时还会连接学院表,把学院的数据也查询出来。这就导致了我们需要为冗余的数据牺牲查询速度。所以解决的思路就是,在统计查询时,仅仅查询我们需要的数据,不查询我们不需要的数据。
解决
在复杂的查询中,往往需要许多关联的数据,可以通过自定义查询和投影的方式,实现在hibernate
中,仅仅查询自己想要的关联数据,提高查询速度。
在上面的例子中,我们可以自定义查询:
select name, score.score from student jion score on score.id=student.id where grade=2019
仅仅查询学生中的名字,学生成绩中的成绩两个信息,之后再通过投影,投影到实体上:
public interface StudengtScoreProjections {
String name;
Double score;
}
整合自定义查询和投影到仓库类中:
public interface StudentRepository extends CrudRepository<Student, Long> {
@Query(value = "select name, score.score as score from student jion score s on score.id=student.id where grade=?1) ",nativeQuery = true)
List<StudengtScoreProjections> getAllByGrade(Long grade);
}
为了复用查询,可以建立响应的视图,查询时从视图中查询:
// 数据库建立视图
create view student_score as
select name, score.score as score,grade from student jion score s on score.id=student.id
// 仓库类中改为从视图中查询
public interface StudentRepository extends CrudRepository<Student, Long> {
@Query(value = "select * from student_score where grade=?1) ",nativeQuery = true)
List<StudengtScoreProjections> getAllByGrade(Long grade);
}
再建立投影和实体之间的转换方法,达到查询的目的。
总结
通过自定义查询和投影的方法,来解决hibernate
自动查询关联数据的问题,使用自定义的查询,可以提高数据库查询的灵活性和速度。
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。