PageHelper 使用的小坑
PageHelper 这个插件用了很多次了,但是今天使用的时候才发现中间有一个小坑,就是比如说,我们一般都是这样进行查询:
//开启分页,page代表当前页数,limit代表每页总数
PageHelper.startPage(page, limit);
//查询数据库的语句,pageHelper在这里做一次拦截器操作
List userInfos = userInfoService.queryAllUserInfo();
//使用PageInfo包装结果
PageInfo pageInfo = new PageInfo<>(userInfos);
上述操作是这样的,但是如果我们需要将查询的 List转换成前端展示的 ViewObject(VO) 对象,那么我们想当然会在中间做一次操作:
PageHelper.startPage(page, limit);
List userInfos = userInfoService.queryAllUserInfo();
//将userInfos某些字段隐藏,转换成前端需要的bean对象
List res = new ArrayList<>();
for (UserInfo info : pageInfo.getList()) {
res.add(convertUserInfoToVO(info));
}
//然后再包装一下前端对象
PageInfo pageInfo = new PageInfo<>(res);
这样获取的 pageInfo.getTotal()一直是当前的 limit 总数,也就是单页的总数 pageSize,而不是查询的数据总条数。所以,我们最好不要在 pageHelper 两个语句中间插入其它语句,而是在后面做一些操作:
//分页查询
PageHelper.startPage(page, limit);
List userInfos = userInfoService.queryAllUserInfo();
PageInfo pageInfo = new PageInfo<>(userInfos);
//转换成前端VO
List res = new ArrayList<>();
for (UserInfo info : pageInfo.getList()) {
res.add(convertUserInfoToVO(info));
}
//那么这样就可以正确返回了!
return new LayuiTableResult(0, "查询成功", pageInfo.getTotal(), res);
同理,我们也不能在上述 userInfoService.queryAllUserInfo() 这个查询的 Service 的内部逻辑中再做多次从数据库查询的操作,这是因为 PageHelper 拦截查询的永远是第一次查询数据库那个语句的查询操作。
自定义 PageHelper 查询
在某些时候,我们可能并不需要 PageHelper 内部的拦截器帮我们在查询的时候进行 limit,在做一些复杂操作时,自定义 PageHelper 分页也是一种可行的方法。
由于 PageHelper 已经提供了 PageInfo 这个包装类供用户调用,所以我们自定义的操作也并不复杂,只需要把计算的参数写入 PageInfo 就可以使用了。
我们以一个用户信息分页为例:
@GetMapping("/queryUsers")
@ResponseBody
public ForumResult getAllUsers(@RequestParam(defaultValue = "1", value = "currPage") Integer currPage,
@RequestParam(defaultValue = "5", value = "pageSize") Integer pageSize) throws ParseException {
//查询后的用户信息(经过一定的复杂操作处理)
List userList = userService.queryAllUsersWithCurrPage(currPage);
//对PageInfo手动分页
Page page = new Page(currPage, pageSize);
page.setTotal(userList.size());
//计算当前需要显示的数据下标起始值
int startIndex = (currPage - 1) * pageSize;
int endIndex = Math.min(startIndex + pageSize, userList.size());
page.addAll(userList.subList(startIndex,endIndex));
//以Page创建PageInfo
PageInfo pageInfo = new PageInfo<>(page);
List res = new ArrayList<>();
for (User user: pageInfo.getList()) {
UserVO vo = convertQuestionToVO(user);
res.add(vo);
}
return userList.size() == 0 ? return new ForumResult(500, "问题数量为零", null) : new ForumResult(200, "查询成功!", res);
}
上述操作就可以实现自定义分页啦!