使用PageHelper分页插件时踩过的坑

前言

起初,在开发过程中完成了一个分页查询的接口。当时测试了一下,分页信息和数据也一并返回了,没有仔细看返回的数据,本以为接口已经大功告成了。然而并没有结束…

后来在测试数据增加多的时候,不妙的事情发生了。分页参数中很明显不对的就是总条数和总页数。下面进行事故的分析

背景

数据库:sqlserver
orm:mybatis

分页查询异常表现

在这里插入图片描述
在这里插入图片描述

bug修复前的代码
  • 数据库查询的数据
    /**
     * 条件分页查询
     *
     * @param condition   查询条件
     * @param currentPage 当前页数
     * @param pageSize    每页显示条数
     * @return BaseVO<PageInfo < FormData>>
     */
    @Override
    public BaseVO<PageInfo<FormData>> formDataList(String condition, Integer currentPage, Integer pageSize, InitSource initSource) {


        PageHelper.startPage(currentPage, pageSize);
        QueryObject queryObject = analysisConditionUtil.queryObject(condition, initSource);
        if (Objects.isNull(queryObject)) {
            return new BaseVO<>(BaseVO.ILLEGAL_PARAM_CODE, "操作方式有误,仅支持 大于,小于,"
                    + "等于,不等于,大于等于,小于等于,在其中,不在其中");
        }

        List<Map<String, Object>> maps = zhongLvDao.getFormDataList(queryObject);
        
        //将查询到的数据的字段名称替换为中文名称
        //【从数据库返回的数据没有直接返回,做了些数据处理,问题就发生在数据处理的时候】
        List<FormData> formDataList = structFormDatalist(initSource, maps);
        PageInfo<FormData> pageInfo = new PageInfo<>(formDataList);

        return new BaseVO<>(BaseVO.SUCCESS_CODE, "查询成功", pageInfo);
    }

  • 数据处理
    /**
     * 将查询到的数据的第三方的英文字段替换为中文字段
     *
     * @param zhongLvSource json中定义的中文字段和第三方英文字段的对应关系
     * @param dataList      List<Map<英文名称, value>>: 查询到的第三方的数据
     * @return List<FormData> 英文字段替换为中文字段的最终数据
     */
    private List<FormData> structFormDatalist(InitSource zhongLvSource, List<Map<String, Object>> dataList) {

       //1.非常不妙的事情其实就发生在这个地方,datalist本身是一个page对象,里面包含了分页信息
       //2.但是经过数据处理后需要返回 List<FormData>的数据结构,就在创建了新的对象的同时并没有将已经存在的分页信息拷贝到新的对象中,导致数据丢失

        List<FormData> formDataList = new ArrayList<>();
        
        for (Map<String, Object> stringDataBaseMap : dataList) {
            FormData formData = new FormData();
            Map<String, DataBase> data = new HashMap<>();

            Set<Map.Entry<String, Object>> entries = stringDataBaseMap.entrySet();

            for (Map.Entry<String, Object> entry : entries) {
                Field fieldByEnName = zhongLvSource.getFieldByEnName(entry.getKey());
                if (!Objects.isNull(fieldByEnName)) {
                    constructCommdata(data, entry, fieldByEnName.getCnName(), fieldByEnName.getType());
                } else {
                    Field fieldByAlias = zhongLvSource.getFieldByAlias(entry.getKey());
                    if (!Objects.isNull(fieldByAlias)) {
                        constructCommdata(data, entry, fieldByAlias.getCnName(), fieldByAlias.getType());
                    }
                }
                formData.setData(data);
            }
            formDataList.add(formData);
        }
        return formDataList;
    }
  • 问题分析
    • 下图为数据库返回的数据进行处理的逻辑,处理完成返回ArrayList类型的数据
      在这里插入图片描述
  • 将返回的数据通过
    在这里插入图片描述
  • 跟踪进入PageInfo方法,追查赋值过程
    在这里插入图片描述
    在这里插入图片描述
修复bug
  • bug修复后的查询结果
    在这里插入图片描述
  • 代码改动的地方
    /**
     * 将查询到的数据的第三方的英文字段替换为中文字段
     *
     * @param zhongLvSource json中定义的中文字段和第三方英文字段的对应关系
     * @param dataList      List<Map<英文名称, value>>: 查询到的第三方的数据
     * @return List<FormData> 英文字段替换为中文字段的最终数据
     */
    private List<FormData> structFormDatalist(InitSource zhongLvSource, List<Map<String, Object>> dataList) {

        //拷贝分页信息【在返回的时候需要保持原来的数据结构,得返回一个新的page对象,并且将分页信息拷贝到新的page对象中】
        Page<FormData> formDataList = new Page<>();
        BeanUtils.copyProperties(dataList, formDataList);

        for (Map<String, Object> stringDataBaseMap : dataList) {
            FormData formData = new FormData();
            Map<String, DataBase> data = new HashMap<>();

            Set<Map.Entry<String, Object>> entries = stringDataBaseMap.entrySet();

            for (Map.Entry<String, Object> entry : entries) {
                Field fieldByEnName = zhongLvSource.getFieldByEnName(entry.getKey());
                if (!Objects.isNull(fieldByEnName)) {
                    constructCommdata(data, entry, fieldByEnName.getCnName(), fieldByEnName.getType());
                } else {
                    Field fieldByAlias = zhongLvSource.getFieldByAlias(entry.getKey());
                    if (!Objects.isNull(fieldByAlias)) {
                        constructCommdata(data, entry, fieldByAlias.getCnName(), fieldByAlias.getType());
                    }
                }
                formData.setData(data);
            }
            formDataList.add(formData);
        }
        return formDataList;

    }
  • 代码修改后再次追踪,发现已经能够正常获取总条数
    在这里插入图片描述
总结

使用pagehelper执行分页时,如果中途需要进行数据处理,切记不能丢失从数据库返回的数据类型和相关数据

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
PageHelper 是一个用于 MyBatis 的分页插件,可以方便地实现数据库查询结果的分页功能。下面是使用 PageHelper 的步骤: 1. 在 Maven 或 Gradle 项目的 `pom.xml`(或 `build.gradle`)文件中添加 PageHelper 依赖: ```xml <!-- Maven --> <dependency> <groupId>com.github.pagehelper</groupId> <artifactId>pagehelper</artifactId> <version>latest-version</version> </dependency> ``` ```groovy // Gradle implementation 'com.github.pagehelper:pagehelper:latest-version' ``` 请确保将 `latest-version` 替换为最新版本号。 2. 修改 MyBatis 配置文件(通常是 `mybatis-config.xml`),在 `<configuration>` 标签下添加 PageHelper 的插件配置: ```xml <plugins> <plugin interceptor="com.github.pagehelper.PageInterceptor"> <property name="dialect" value="your-database-dialect"/> <!-- 其他配置属性 --> </plugin> </plugins> ``` 这里的 `your-database-dialect` 需要根据你使用的数据库类型来设置,例如对于 MySQL 可以设置为 `mysql`。 3. 在需要分页查询的方法中加入分页参数,并调用 PageHelper 的静态方法 `startPage` 来开启分页功能。 ```java import com.github.pagehelper.PageHelper; public List<User> getUsers(int pageNum, int pageSize) { // 开始分页pageNum 表示当前页码,pageSize 表示每页显示的记录数 PageHelper.startPage(pageNum, pageSize); // 查询数据库,获取结果列表 List<User> users = userDao.getUsers(); return users; } ``` 这样,PageHelper 就会自动在查询语句中添加分页的 SQL 语句,完成分页查询的功能。 请注意,PageHelper 还提供了很多其他的配置和使用方式,例如设置排序,自定义分页参数等,你可以根据具体需求参考官方文档进行配置和使用

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值