Mybatis中分页PageHelper的使用-PageHelper插件,无故给SQL增加Limit问题-《Java笔记》

admin 2025-10-19 04:09:50 编程 来源:ZONE.CI 全球网 0 阅读模式

    Java Mybatis PageHelper在项目中,使用了 MyBatis 的 PageHelper 插件,发现了一个奇怪的问题,经常会给SQL无缘无故的增加Limit语句,经过调查,发现原因是没有安全的使用PageHelper插件,先来看一个例子:

    1. PageHelper.startPage(pageable.getPageNumber(), pageable.getPageSize());
    2. PageInfolist = null;
    3. if (condition.getCustomerIdListForUser() != null && condition.getCustomerIdListForUser().size() > 0) {
    4. list = new PageInfo<>(orderMapper.findOrderList(condition)); //分页查询语句
    5. } else {
    6. list = new PageInfo<>(new ArrayList());
    7. list.setPageNum(0);
    8. list.setPageSize(1);
    9. list.setTotal(0);
    10. }

    在例子中,PageHelper.startPage就属于明显的不安全调用,因为PageHelper的原理是,在PageHelper.startPage调用时,会给全局对象LOCAL_PAGE设值,然后通过拦截器拦截每个SQL语句,如果LOCAL_PAGE有值,则给该SQL增加Limit,并调用clearPage方法清除LOCAL_PAGE的值;但是上面的代码,其分页查询语句有可能因为if的条件的不满足没有执行,所以在程序执行结束时,PageHelper.startPage已经执行,LOCAL_PAGE的值已经设置当线程池再次分配该线程执行其他程序时,可能会在该程序的第一个SQL上增加了Limit语句。解决该问题的方法是,要绝对保证PageHelper.startPage和分页查询语句之间不要有任何其他语句,或者在程序结束时增加PageHelper.clearPage();的调用,例:

    1. PageInfolist = null;
    2. if (condition.getCustomerIdListForUser() != null && condition.getCustomerIdListForUser().size() > 0) {
    3. PageHelper.startPage(pageable.getPageNumber(), pageable.getPageSize());
    4. list = new PageInfo<>(orderMapper.findOrderList(condition));//分页查询语句
    5. } else {
    6. list = new PageInfo<>(new ArrayList());
    7. list.setPageNum(0);
    8. list.setPageSize(1);
    9. list.setTotal(0);
    10. }

    1. PageHelper.startPage(pageable.getPageNumber(), pageable.getPageSize());
    2. PageInfolist = null;
    3. if (condition.getCustomerIdListForUser() != null && condition.getCustomerIdListForUser().size() > 0) {
    4. list = new PageInfo<>(orderMapper.findOrderList(condition));
    5. } else {
    6. list = new PageInfo<>(new ArrayList());
    7. list.setPageNum(0);
    8. list.setPageSize(1);
    9. list.setTotal(0);
    10. }
    11. PageHelper.clearPage();

    另外也可以这样

    1. PageHelper.startPage(pageable.getPageNumber(), pageable.getPageSize());
    2. PageInfolist = null;
    3. if (condition.getCustomerIdListForUser() != null && condition.getCustomerIdListForUser().size() > 0) {
    4. try{
    5. list = new PageInfo<>(orderMapper.findOrderList(condition));
    6. }finally{
    7. PageHelper.clearPage();
    8. }
    9. } else {
    10. list = new PageInfo<>(new ArrayList());
    11. list.setPageNum(0);
    12. list.setPageSize(1);
    13. list.setTotal(0);
    14. }

    但是这样写有点冗余,而且没有必要。官网的解释 :PageHelper 方法使用了静态的 ThreadLocal 参数,分页参数和线程是绑定的。只要可以保证在 PageHelper 方法调用后紧跟 MyBatis 查询方法,这就是安全的。因为 PageHelper 在 finally 代码段中自动清除了 ThreadLocal 存储的对象。如果代码在进入 Executor 前发生异常,就会导致线程不可用,这属于人为的 Bug(例如,接口方法和 XML 中的不匹配,导致找不到 MappedStatement 时), 这种情况由于线程不可用,也不会导致 ThreadLocal 参数被错误的使用。

    以太坊cppgolang区别 编程

    以太坊cppgolang区别

    以太坊是一种去中心化的开源平台,它采用智能合约技术,旨在构建和运行不受干扰的分布式应用程序。作为目前最受欢迎的区块链平台之一,以太坊提供了多种编程语言的支持,其
    progolang 编程

    progolang

    Go语言(Golang)是由Google开发的一门静态类型编程语言。作为一名专业的Golang开发者,我深知这门语言的优势和特点。在本文中,我将介绍Golang
    golangn个发送者 编程

    golangn个发送者

    Golang是一种开源的编程语言,由Google团队开发,旨在提高程序的并发性和简化软件开发过程。在Go语言中,有时需要向多个接收者发送信息。本文将介绍如何在G
    golang技能图谱 编程

    golang技能图谱

    从互联网行业的快速发展到人工智能技术的日益成熟,各种编程语言也应运而生。而在这众多的编程语言中,Golang(即Go)作为一门强大且高效的开发语言备受关注。Go
    评论:0   参与:  5