业务:查询功能,需要关联多张表,为一对多或多对多。
需求:要求分页,分页统计总数。
使用技术: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解决。
以下说明我的错误经历
/**
* 这里的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,否则报异常。
/**
* 这里的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,干扰了分页查询。
/**
* 这里的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。
猜测原因:可能使用的是两个不同的分页对象,所以第二次只统计了总数,没有检测到没有分页查询
/**
* 这里的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);
经过验证,最后一种方式也是有问题的,此时既可以分页,也可以统计数量,但是分页的结果还是最小单元。建议使用方式2,SQL使用子查询来先分页后关联数据