PageHelper中分页失效的原因分析与正确方法实践

 更新时间:2025年11月20日 09:06:59   作者:秧歌star519  
在使用 PageHelper 插件开发查询接口时,有时会出现分页失效的情况,下面小编就和大家详细讲讲分页失效的原因分析与正确方法,希望对大家有所帮助

1. 问题现象

在使用 PageHelper 插件开发查询接口时,出现分页失效的情况:接口返回了全部数据而非当前页数据,且 PageInfo 对象中的分页元数据(如 total 总条数、pages 总页数)计算错误。

错误代码示例

public BaseResponse<MyDataDTO> queryDataList(int pageNum, int pageSize, Map<String, Object> params) {
    BaseResponse response = new BaseResponse();
    
    // 1. 错误:在设置分页参数前执行了数据库查询
    List<MyDataDTO> dataList = myMapper.selectData(params);
    
    // 2. 判空逻辑
    if (!CollectionUtils.isEmpty(dataList)) {
        // 3. 错误:查询早已完成,此时调用 startPage 无效
        PageHelper.startPage(pageNum, pageSize);
        
        // 4. 错误:直接使用全量 List 包装 PageInfo,无法获取数据库真实总数
        PageInfo<MyDataDTO> pageInfo = new PageInfo<>(dataList);
        
        response.setResult(wrapPageResult(pageInfo));
        return response;
    }
    
    response.setResult(Collections.EMPTY_LIST);
    return response;
}

2. 原理分析

PageHelper 的核心机制基于 ThreadLocalMyBatis 拦截器(Interceptor)

2.1 执行流程

PageHelper 不是在内存中对结果集进行截取,而是通过拦截器修改 SQL 语句。

设置参数:调用 PageHelper.startPage(...) 时,插件会将分页参数(pageNum, pageSize)存入当前线程的 ThreadLocal 中。

拦截 SQL:当 MyBatis 执行 Mapper 方法时,PageHelper 拦截器会触发。

SQL 改写

  • 拦截器检查 ThreadLocal 中是否存在分页参数。
  • 若存在:拦截器会根据数据库方言(如 MySQL)生成 SELECT COUNT(0) 语句获取总数,并将原 SQL 改写为带 LIMIT/OFFSET 的分页 SQL 执行。
  • 若不存在:拦截器直接放行,执行原始 SQL。

清理上下文:SQL 执行结束后,拦截器会清除 ThreadLocal 中的分页参数,避免污染后续查询。

2.2 失效原因

在上述错误代码中:

  • 执行顺序错误myMapper.selectDataPageHelper.startPage 之前执行。
  • 拦截失败:执行查询时,ThreadLocal 中没有任何分页参数,拦截器未生效,MyBatis 执行了全量查询。
  • 参数无效:查询结束后才调用 startPage,虽然设置了 ThreadLocal,但 SQL 交互已结束,该参数未被消费。
  • 元数据错误PageInfo 接收的是全量 List,因此它只能基于 List 的大小计算 total,导致分页信息不符合预期。

3. 正确实现

核心原则PageHelper.startPage 必须紧邻 Mapper 查询方法之前调用。

修正代码

public BaseResponse<MyDataDTO> queryDataList(int pageNum, int pageSize, Map<String, Object> params) {
    BaseResponse response = new BaseResponse();

    // 1. 设置分页参数(存入 ThreadLocal)
    PageHelper.startPage(pageNum, pageSize);
    
    // 2. 执行查询(拦截器生效,自动改写 SQL 并执行 Count 查询)
    // 注意:此时返回的 list 实际类型为 Page<E>
    List<MyDataDTO> dataList = myMapper.selectData(params);

    // 3. 获取分页结果
    PageInfo<MyDataDTO> pageInfo = new PageInfo<>(dataList);
    
    // 4. 结果处理(PageInfo 可安全处理空集合)
    if (!CollectionUtils.isEmpty(dataList)) {
         // pageInfo.getTotal() 为数据库真实总数
         response.setResult(wrapPageResult(pageInfo));
    } else {
         response.setResult(Collections.EMPTY_LIST);
    }
    
    return response;
}

4. 最佳实践与注意事项

严格遵守调用顺序 必须保证 startPage -> Mapper查询 -> PageInfo包装 的执行顺序。

避免逻辑穿插 严禁在 startPageMapper查询 之间插入其他 SQL 操作或复杂逻辑。

风险:PageHelper 的分页参数是“一次性消费”的。如果在分页查询前插入了其他 SQL(如查询用户信息),分页参数会被那条 SQL 消费掉,导致原本需要分页的主查询失效。

PageInfo 的健壮性 无需为了判空调整代码顺序。PageInfo 对空 List 有良好的兼容性,若查询结果为空,它会自动设置 total=0,不会抛出异常。

大数据量风险 如果因顺序错误导致分页失效,全量查询可能会将百万级数据加载至内存,极易引发 OOM(内存溢出),影响系统稳定性。

到此这篇关于PageHelper中分页失效的原因分析与正确方法实践的文章就介绍到这了,更多相关PageHelper分页失效解决内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java开发之Spring连接数据库方法实例分析

    Java开发之Spring连接数据库方法实例分析

    这篇文章主要介绍了Java开发之Spring连接数据库方法,以实例形式较为详细的分析了Java Spring开发中针对数据库的相关操作技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-10-10
  • Java并发框架中的AQS详细解析

    Java并发框架中的AQS详细解析

    这篇文章主要介绍了Java并发框架中的AQS详细解析,之前说锁的升级的时候,说到了自旋锁会空转几次尝试等待获取资源,其实这一系列的动作是有一个规范的这个规范叫做同步发生器AbstractQueuedSynchronizer ,简称AQS,需要的朋友可以参考下
    2024-01-01
  • Java正则表达式的实例操作指南

    Java正则表达式的实例操作指南

    这篇文章主要给大家介绍了关于Java正则表达式的实例操作指南,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-09-09
  • 浅谈Java动态代理的实现

    浅谈Java动态代理的实现

    最近,小组同事做代码改造时,使用到了动态代理,自己阅读时,发现对代理这种设计模式都不怎么清楚,导致理解代码也很困难 自己唯一能看懂的,大概就是handler中的invoke方法 ,文中作出了非常详细的介绍,需要的朋友可以参考下
    2021-05-05
  • SpringBoot3整合Hutool-captcha实现图形验证码

    SpringBoot3整合Hutool-captcha实现图形验证码

    在整合技术框架的时候,想找一个图形验证码相关的框架,看到很多验证码的maven库不再更新了或中央仓库下载不下来,还需要多引入依赖,后面看到了Hutool图形验证码(Hutool-captcha)中对验证码的实现,所以本文介绍了SpringBoot3整合Hutool-captcha实现图形验证码
    2024-11-11
  • Spring中propagation的传播机制详解

    Spring中propagation的传播机制详解

    这篇文章主要介绍了Spring中propagation的传播机制详解,要搞懂事务的传播机制,那么就要明白逻辑事务中各个事务的关系,才能彻底理解事务传播特性,在Spring事务中,各个逻辑事务的关系可以是并列、覆盖或包含,需要的朋友可以参考下
    2023-12-12
  • Java多维数组和Arrays类方法总结详解

    Java多维数组和Arrays类方法总结详解

    这篇文章主要介绍了Java多维数组和Arrays类方法总结详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-03-03
  • 详解Java的Proxy动态代理机制

    详解Java的Proxy动态代理机制

    Java有两种代理方式,一种是静态代理,另一种是动态代理。对于静态代理,其实就是通过依赖注入,对对象进行封装,不让外部知道实现的细节。很多 API 就是通过这种形式来封装的
    2021-06-06
  • Java的Lock接口与读写锁详解

    Java的Lock接口与读写锁详解

    这篇文章主要介绍了Java的Lock接口与读写锁详解,锁是用来控制多个线程访问共享资源的方式,一般来说,一个锁能够防止多个线程同时访问共享资源,但是有些锁可以允许多个线程并发的访问共享资源,比如读写锁,需要的朋友可以参考下
    2023-12-12
  • Java中通过Class类获取Class对象的方法详解

    Java中通过Class类获取Class对象的方法详解

    这篇文章主要给大家介绍了关于Java中通过Class类获取Class对象的方法,文中通过示例代码介绍的非常详细,对大家学习或者使用java具有一定的参考学习价值,需要的朋友们下面跟着小编来一起学习学习吧。
    2017-08-08

最新评论