Mybatis-plus分页查询不生效问题排查全过程

 更新时间:2023年03月15日 11:24:09   作者:追寻上飞  
最近写分页的时候,遇到了分页无法正常发挥作用的问题,下面这篇文章主要给大家介绍了关于Mybatis-plus分页查询不生效问题排查的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下

一、问题描述

在查询的时候,发现点击后台的分页器数字,第2页时候,数据还是和第1页的一致。就看后台的数据库打印语句,如下所示。点击第一页和第二页都是这个,limit后的参数只有一个,前期做过类似,点击第二页分页的时候,语句是LIMIT ?,?

ON ap.id = a.project_id ORDER BY a.create_time DESC LIMIT ?

二、分析步骤

1.首先开始怀疑的是自己的分页对象出现了问题,因为MySQL ORM框架使用了JPA框架遗留的代码。将spring-data的分页对象org.springframework.data.domain.Pageable转成了mybatis-plus的分页对象com.baomidou.mybatisplus.extension.plugins.pagination.Page<T>

debug后,Page<T>的 current 和 size 都是存在且对应前台传来的值。

2.接着怀疑是mybatis-plus的拦截器顺序问题,因为项目里写了数据权限的拦截器,在研究数据权限拦截器的时候就看到有说拦截器添加顺序会影响到SQL语句拼接的正确性。对比了正常的添加顺序后,这部分也没有问题。

   @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor() {
        MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();

        //添加数据权限处理器,注意顺序,先数据权限再分页
        MyDataPermissionInterceptor dataPermissionInterceptor = new MyDataPermissionInterceptor();
        // 添加自定义的数据权限处理器
        dataPermissionInterceptor.setDataPermissionHandler(new MyDataPermissionHandler());
        interceptor.addInnerInterceptor(dataPermissionInterceptor);

        interceptor.addInnerInterceptor(new PaginationInnerInterceptor());
        return interceptor;
    }

3.发现这两部分都没问题后,决定还是debug一下分页拦截器。

重要的分页SQL语句拼接函数如下。

DialectModel model = dialect.buildPaginationSql(buildSql, page.offset(), page.getSize());

进入到里面,这里数据库使用的MySQL。

public class MySqlDialect implements IDialect {

    @Override
    public DialectModel buildPaginationSql(String originalSql, long offset, long limit) {
        StringBuilder sql = new StringBuilder(originalSql).append(" LIMIT ").append(FIRST_MARK);
        if (offset != 0L) {
            sql.append(StringPool.COMMA).append(SECOND_MARK);
            return new DialectModel(sql.toString(), offset, limit).setConsumerChain();
        } else {
            return new DialectModel(sql.toString(), limit).setConsumer(true);
        }
    }
}

可以看到偏移量offset至关重要,决定了拼接的条件判断。

又发现,offet的值和current相关,根据排查结果来看,current 是正常传值的。

default long offset() {
        long current = this.getCurrent();
        return current <= 1L ? 0L : Math.max((current - 1L) * this.getSize(), 0L);
    }

因此到这里我们就会发现,是offset的真实值不正确。

当后台前端页面传来的current0时,offset等于0

当后台前端页面传来的current1时,offset依旧等于0

而回到拼接函数buildPaginationSql中,不等于0的时候才会有 两个参数的拼接。

至此,这个问题解决了,就是在对象转换时,没有对current的值进行 + 1

三、解决方案

public class MyPage<T> extends Page<T> {

    /**
     * @Description: 将spring的分页对象转成Mybatis
     * @param pageable:
     * @return: null
     **/
    public MyPage(Pageable pageable) {
        this.setSize(pageable.getPageSize());
        this.setCurrent(pageable.getPageNumber() + 1);
    }
}

四、总结

  • 多用编译器的debug
  • 细节问题还是要注意,哪怕一个变量

到此这篇关于Mybatis-plus分页查询不生效问题排查的文章就介绍到这了,更多相关Mybatis-plus分页查询不生效内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 自带IDEA插件的阿里开源诊断神器Arthas线上项目BUG调试

    自带IDEA插件的阿里开源诊断神器Arthas线上项目BUG调试

    这篇文章主要为大家介绍了自带IDEA插件阿里开源诊断神器Arthas线上项目BUG调试,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • java实现用户自动登录

    java实现用户自动登录

    这篇文章主要为大家详细介绍了java用户自动登录的实现方法,分为六个步骤实现用户自动登录,并验证用户是否已经登录,感兴趣的小伙伴们可以参考一下
    2016-03-03
  • Java8时间转换(LocalDateTime)代码实例

    Java8时间转换(LocalDateTime)代码实例

    这篇文章主要介绍了java8时间转换(LocalDateTime)代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-11-11
  • 详解Java七大阻塞队列之SynchronousQueue

    详解Java七大阻塞队列之SynchronousQueue

    SynchronousQueue不需要存储线程间交换的数据,它的作用像是一个匹配器,使生产者和消费者一一匹配。本文详细讲解了Java七大阻塞队列之一SynchronousQueue,需要了解的小伙伴可以参考一下这篇文章
    2021-09-09
  • 基于springMvc+hibernate的web application的构建

    基于springMvc+hibernate的web application的构建

    下面小编就为大家带来一篇基于springMvc+hibernate的web application的构建。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-10-10
  • 一个简易的Java多页面队列爬虫程序

    一个简易的Java多页面队列爬虫程序

    这篇文章主要为大家详细介绍了一个多页面的java爬虫,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-08-08
  • Spring Data JPA映射自定义实体类操作

    Spring Data JPA映射自定义实体类操作

    这篇文章主要介绍了Spring Data JPA映射自定义实体类操作,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-11-11
  • 使用list stream: 任意对象List拼接字符串

    使用list stream: 任意对象List拼接字符串

    这篇文章主要介绍了使用list stream:任意对象List拼接字符串操作,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-09-09
  • Spring Cloud根据服务名获取服务的ip端口问题

    Spring Cloud根据服务名获取服务的ip端口问题

    这篇文章主要介绍了Spring Cloud根据服务名获取服务的ip端口,本篇示例我就以Nacos注册中心为例了,下面是我注册的两个服务,需要的朋友可以参考下
    2022-09-09
  • springboot中设置定时任务的三种方法小结

    springboot中设置定时任务的三种方法小结

    在我们开发项目过程中,经常需要定时任务来帮助我们来做一些内容,本文介绍了springboot中设置定时任务的三种方法,主要包括@Scheduled注解,Quartz框架和xxl-job框架的实现,感兴趣的可以了解一下
    2023-12-12

最新评论