1

数据库查询一直是提高程序速度的瓶颈之一,本人也遇到了因为数据库数据查询过久导致影响失败,在这里记录以下解决的方法。

描述

该功能为统计功能,根据参数查找出符合的器具信息,然后再根据统计信息得出统计数据。就好比要统计某个年级学生的优秀,良好,及格率。看起来简单容易实现,但是到了测试环境中,却出现了错误:查询过久得不到响应。

排查

经过打断点查看,发现系统问题出在了查询数据的过程中。经过排查是数据过大,并且要连接的表过多,导致查询时间花费过大,前台得不到响应。

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自动查询关联数据的问题,使用自定义的查询,可以提高数据库查询的灵活性和速度。


鲸冬香
456 声望27 粉丝

引用和评论

0 条评论