PageHelper关联查询 统计总数问题

Posted £漫步 云端彡

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了PageHelper关联查询 统计总数问题相关的知识,希望对你有一定的参考价值。

项目场景:

业务:查询功能,需要关联多张表,为一对多或多对多。
需求:要求分页,分页统计总数。
使用技术:Mybatis,PageHelper


问题描述

正常查询一张表的情况下,官方推荐

//获取第1页,10条内容,默认查询总数count
PageHelper.startPage(1, 10);
List<User> list = userMapper.selectAll();
//用PageInfo对结果进行包装
PageInfo page = new PageInfo(list);

但是这种方式实际上PageHelper会自动生成:

select count(0) from (selectAll() sql语句) table_count

普通查询返回正常统计数量。一对多、多对多关联返回的count不一定正确


原因分析:

关联查询的结果是以最小单位为一条数据进行统计的。这里的最小单位举例:学生表和成绩表关联,一个学生有多个课程成绩,那么查询学生的成绩时,一位学生+一门课程的成绩就为最小单位。但有时候我们需要根据成绩搜索学生,但是需要了解有多少个学生。
这个时候PageHelper就无法自动实现了,需要自定义count解决。


解决方案:

以下说明我的错误经历

  • 错误方式1:
/**
 * 这里的getUserList_COUNT SQL模板是:
 * SELECT count(*) FROM (getUserList SQL语句) table_count
 * 返回值类型:Long
 */
// false 表示不使用统计总数,自定义统计数量
PageHelper.startPage(1, 10, false).doCount(() -> stopShipMapper.getUserList_COUNT(params));
// params这里指的是自定义的参数对象,可有可无,业务需求而已
List<User> userList = stopShipMapper.getUserList(params);
PageInfo<User> info = new PageInfo<>(userList);

报错:在系统中发现了多个分页插件,请检查系统配置!
网上多数介绍,通过@SpringBootApplication(exclude = PageHelperAutoConfiguration.class),但是分页会失效,然后还有说多了两次自定义配置。但是只进行了一次配置。不存在这个问题。
查看源码得知如下:

原因:也就是说查询方法名不能后缀_COUNT,否则报异常。

  • 错误方式2:
    于是乎改了调用方法名:
/**
 * 这里的getUserList_COUNT SQL模板是:
 * SELECT count(*) FROM (getUserList SQL语句) table_count
 * 返回值类型:Long
 */
// false 表示不使用统计总数,自定义统计数量
PageHelper.startPage(1, 10, false).doCount(() -> stopShipMapper.getUserListTotal(params));
// params这里指的是自定义的参数对象,可有可无,业务需求而已
List<User> userList = stopShipMapper.getUserList(params);
PageInfo<User> info = new PageInfo<>(userList);

错误:谢天谢地,count统计正常,但是查询语句后面没有后缀limit,分页失效。

原因:PageHelper分页有效的SQL为PageHelper.startPage后紧跟着的一条SQL。上述代码紧跟着的SQL是count,干扰了分页查询。

  • 错误方式3:
    然后进一步修改,得到如下代码
/**
 * 这里的getUserList_COUNT SQL模板是:
 * SELECT count(*) FROM (getUserList SQL语句) table_count
 * 返回值类型:Long
 */
// false 表示不使用统计总数,自定义统计数量
PageHelper.startPage(1, 10, false);
// params这里指的是自定义的参数对象,可有可无,业务需求而已
List<User> userList = stopShipMapper.getUserList(params);
PageHelper.count(() -> stopShipMapper.getUserListTotal(params))
PageInfo<User> info = new PageInfo<>(userList);

错误:然后又错啦,报错说什么要求return long 实际 return null。

猜测原因:可能使用的是两个不同的分页对象,所以第二次只统计了总数,没有检测到没有分页查询

  • 错误方式4:
    接下来再改,得到如下代码
/**
 * 这里的getUserListTotal SQL模板是:
 * SELECT count(*) FROM (getUserList SQL语句) table_count
 * 返回值类型:Long
 */
// false 表示不使用统计总数,自定义统计数量
// false 表示不使用统计总数
Page page = PageHelper.startPage(1, 10, false);
// params这里指的是自定义的参数对象,可有可无,业务需求而已
List<User> userList = stopShipMapper.getUserList(params);
// 自定义统计数量
page.doCount(()-> stopShipMapper.getUserListTotal(params));
PageInfo<User> info = new PageInfo<>(userList);

错误:分页数据正常啦,但是统计count又成了-1。

原因:doCount是一个回调自定义处理函数,并没有对page做处理,因此需要手动处理。

  • 最终正确方式:
/**
 * 这里的getUserListTotal SQL模板是:
 * SELECT count(*) FROM (getUserList SQL语句) table_count
 * 返回值类型:Long
 */
// false 表示不使用统计总数,自定义统计数量
// false 表示不使用统计总数
Page page = PageHelper.startPage(1, 10, false);
// params这里指的是自定义的参数对象,可有可无,业务需求而已
List<User> userList = stopShipMapper.getUserList(params);
// 自定义统计数量
page.doCount(() -> page.setTotal(stopShipMapper.getUserListTotal(params)));
PageInfo<User> info = new PageInfo<>(userList);

以上是关于PageHelper关联查询 统计总数问题的主要内容,如果未能解决你的问题,请参考以下文章

PageHelper关联查询 统计总数问题

pagehelper怎么计算总数的

记录pageHelper分页orderby的坑

怎样用ASP统计查询记录的总数

Mybatisplus 分页查询时,禁止自动统计总数

Rails,如何避免对关联中的总数(计数、大小、counter_cache)进行“N + 1”查询?