PageHelper自动增加limit产生的数据查询问题

今天线上项目跑着跑着,老是查询不到正确的数据,后面对sql语句进行打印分析,发现有些时候mybatits就算使用了selectByPrimary也会
自动给我加上 limit ,后面查询资料发现,是我 PageHelper.startPage(1, 10); 和mybatits查询代码之间还有别的代码,分页参数和线程是绑定的,如果本次没有去查询数据,该线程去执行别的sql语句时就会自动加上limit ,具体方法请看开发者给出的说明文档。

总之,如果设置分页条件和sql查询语句之间还有别的代码的话就只能自己手动调用清除线程中分页参数缓存信息,

PageHelper.clearPage();

PageHelper 安全调用

  • 使用 RowBounds 和 PageRowBounds 参数方式是极其安全的

  • 使用参数方式是极其安全的

  • 使用 ISelect 接口调用是极其安全的

  • ISelect 接口方式除了可以保证安全外,还特别实现了将查询转换为单纯的 count 查询方式,这个方法可以将任意的查询方法,变成一个 select count(*) 的查询方法。

  • 什么时候会导致不安全的分页?

PageHelper 方法使用了静态的 ThreadLocal 参数,分页参数和线程是绑定的。

只要你可以保证在 PageHelper 方法调用后紧跟 MyBatis 查询方法,这就是安全的。因为 PageHelper 在 finally 代码段中自动清除了 ThreadLocal 存储的对象。

如果代码在进入 Executor 前发生异常,就会导致线程不可用,这属于人为的 Bug(例如接口方法和 XML 中的不匹配,导致找不到 MappedStatement 时), 这种情况由于线程不可用,也不会导致 ThreadLocal 参数被错误的使用。

但是如果你写出下面这样的代码,就是不安全的用法:

PageHelper.startPage(1, 10);
List<Country> list;
if(param1 != null){
    list = countryMapper.selectIf(param1);
} else {
    list = new ArrayList<Country>();
}

这种情况下由于 param1 存在 null 的情况,就会导致 PageHelper 生产了一个分页参数,但是没有被消费,这个参数就会一直保留在这个线程上。当这个线程再次被使用时,就可能导致不该分页的方法去消费这个分页参数,这就产生了莫名其妙的分页。

上面这个代码,应该写成下面这个样子:

List<Country> list;
if(param1 != null){
    PageHelper.startPage(1, 10);
    list = countryMapper.selectIf(param1);
} else {
    list = new ArrayList<Country>();
}

这种写法就能保证安全。

如果你对此不放心,你可以手动清理 ThreadLocal 存储的分页参数,可以像下面这样使用:

List<Country> list;
if(param1 != null){
    PageHelper.startPage(1, 10);
    try{
        list = countryMapper.selectAll();
    } finally {
        PageHelper.clearPage();
    }
} else {
    list = new ArrayList<Country>();
}
  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
根据引用所提到的情况,如果在使用Spring Boot项目中的PageHelper进行分页查询时,没有在SQL语句中拼接上limit,导致查询出全部数据。解决这个问题的方法是在调用Mapper时直接添加PageHelper.startPage,而不是在其他调用层添加。这样可以确保PageHelper的分页功能生效。引用中也提到了对于PageHelper.startPage的使用,它只对接下来的第一个SQL语句执行自动分页,自动添加limit语句。所以,在调用Mapper时添加PageHelper.startPage可以确保分页生效。另外,根据引用所提到的,在不同的框架中,对于jar包的依赖和PageHelper的配置可能会有所不同。在Spring MVC框架中,需要配置PageHelper来使用其分页功能。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [pageHelper不生效,sql没有自动加上limit](https://blog.csdn.net/qq_46095164/article/details/125972196)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [pagehlper不生效,sql没有自动加上limit](https://blog.csdn.net/big_bigwolf/article/details/98625694)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值