MyBatis中使用第三方分页插件PageHelper完成分页功能

news/2024/10/30 15:22:13/

文章目录

  • 一、前言
  • 二、基于插件拦截方式
    • 1、自定义插件
    • 2、使用第三方插件完成分页
      • 1、分页插件的配置
      • 2、分页插件的使用

一、前言

分页是web应用程序非常重要的一个技术。数据库中的数据可能是成千上万的,不可能把这么多的数据一次显示在浏览器上面。一般根据每行数据在页面上所占的空间每页显示若干行,比如一般20行是一个比较理想的显示状态。
在这里插入图片描述

对于MyBatis中分页的操作其实是有两种,一种是基于参数的改造, 还有一种是基于插件的拦截。

对于参数的改造,可以直接使用通过添加参数limitoffset就可以实现查询从某个位置开始的若干记录,代码实现如下所示:

  <select id="selectSomeData" parameterType="map" resultType="com.somedata">SELECT * FROM sometableORDER BY somecolumnLIMIT #{limit} OFFSET #{offset}</select>

这段 SQL 语句会返回从偏移量为 offset 的位置开始的limit条结果。例如:LIMIT 30,10 表示从第 31 行开始返回10行结果。

在实际应用中,我们可以将limitoffset 抽取成两个参数,并传入到 MyBatis
这种方法在这里就不怎么介绍了,这里主要介绍的是通过基于插件拦截的方式解决分页问题

二、基于插件拦截方式

1、自定义插件

我们可以通过插件的方式来实现, 这种方式更加灵活,支持实现更为复杂的分页功能。我们需要自定义一个拦截器,实现Interceptor接口,并重写其中唯一的intercept方法,在其中对 SQL 语句进行修改,添加分页信息。具体操作如下,首先自己定义一个拦截器

下面写的全是伪代码,不能直接运行的,这里只是为了引出pagehelper

public class PageInterceptor implements Interceptor {/*** 拦截方法     *     * @param invocation     * @return     * @throws Throwable*/@Overridepublic Object intercept(Invocation invocation) throws Throwable {// 获取原始的SQL语句String sql = (String) invocation.getArgs()[0];// 查询总数并计算出总页数和当前页int total = count(sql);// 如果总数小于等于0,则直接返回空结果集if (total <= 0) {return Collections.emptyList();}// 计算出当前页的起始位置和结束位置int offset = getOffset(pageNo, pageSize);int limit = pageSize;// 构造含分页信息的新SQLString newSql = getNewSql(sql, offset, limit);// 将新SQL替换成原来的SQL,并继续执行原有方法ReflectionUtils.setFieldValue(invocation, "h.sql", newSql);Object result = invocation.proceed();// 包装成Page对象,并返回Page<T> pageResult = new Page<>(pageNo,pageSize,total,(List<T>)result);return pageResult;}/*** @author Jeff Fong* @description 合成新的sql * @date 2023/5/22 14:03 * @param: sql* @param: offset* @param: limit* @return java.lang.String**/private String getNewSql(String sql, int offset, int limit) {       return sql + " LIMIT " + offset + "," + limit;    }/*** @author Jeff Fong* @description 获取总的条数* @date 2023/5/22 14:03* @param: sql* @return int**/private int count(String sql){        // code omittedreturn 1;}/*** @author Jeff Fong* @description 计算出当页的offset* @date 2023/5/22 14:02* @param: pageNo* @param: pageSize* @return int**/private int getOffset(int pageNo, int pageSize) {        return (pageNo - 1) * pageSize;    }
}

然后,我们需要在 mybatis-config.xml 配置文件中注册该拦截器:

<plugins>        <plugin interceptor="com.example.mybatis.PageInterceptor"/></plugins>

最后,在业务代码中就可以直接使用

 public List<User> selectUserListByPage(int startRow, int pageSize){RowBounds rowBounds = new RowBounds(startRow,pageSize);String statement = "com.example.UserMapper.selectUserList";return sqlSession.selectList(statement,null,rowBounds);}

2、使用第三方插件完成分页

这里使用的插件是PageHelper, 在https://pagehelper.github.io/docs/howtouse/有关于这个插件的介绍。

1、分页插件的配置

(1)、添加依赖

<!-- https://mvnrepository.com/artifact/com.github.pagehelper/pagehelper -->
<dependency><groupId>com.github.pagehelper</groupId><artifactId>pagehelper</artifactId><version>5.2.0</version>
</dependency>

(2)、配置分页插件
在MyBatis的核心配置文件中配置插件

<plugins>
<!--设置分页插件--><plugin interceptor="com.github.pagehelper.PageInterceptor"></plugin>
</plugins>

一定要注意plugins分页插件在mybatis-config.xml的位置,位置如果不对的话,是会直接报错的。
在这里插入图片描述

2、分页插件的使用

在需要分页的数据之前使用PageHelper.startPage(int pageNum, int pageSize)开启分页功能

  • pageNum:当前页的页码
  • pageSize:每页显示的条数

在查询获取list集合之后,使用PageInfo pageInfo = new PageInfo<>(List list, int navigatePages)获取分页相关数据。

  • list:分页之后的数据
  • navigatePages:导航分页的页码数

看看具体的代码实现,对于mapper中的接口比较简单,这里就不一一的说明;

    /*** limit    index,pagesize* index    当前页的起始索引* pageSize 每页显示的条数* pageNum  当前页的页码* 当前页的起始索引 = 每页条数 * 页码 - 1* index = pageNum * pageSize - 1** 通过索引获得数据** 使用MyBatis的分页插件,实现分页功能:* 1。需要在查询功能之前开启分页* PageHelper.startPage(2, 4);** 2。在查询功能之后获取分页相关信息*   PageInfo<Emp> pages = new PageInfo<>(emps, 5); 5表示导航分页的数量*/@Testpublic void test02() throws IOException {SqlSession sqlSession = getSqlSession();UserMapper mapper = sqlSession.getMapper(UserMapper.class);System.out.println("\n查询功能前启用插件.....");PageHelper.startPage(2, 4);List<User> all = mapper.findAll();all.forEach(System.out::println);System.out.println("\n");PageInfo<User> pages = new PageInfo<User>(all,);System.out.println("-------->" + pages);}

实际上的所有数据
在这里插入图片描述
查询第二页的数据,应该是
在这里插入图片描述
最后来看看最终的返回的详细数据:
在这里插入图片描述
看到所需要的数据,都在pageinfo对象中
首先来提一嘴
在这里插入图片描述
这里的navigatepages是什么东西?
就是相当于前端要显示的下面导航栏的东西
在这里插入图片描述
这里的navigatepages = [1,2,3,4,5....., 11]



http://www.ppmy.cn/news/84181.html

相关文章

springboot 上传文件常见问题汇总

springboot 上传文件常见问题汇总 AccessDeniedException步骤配置怎么确保应用程序有这个路径及其子路径的访问权限步骤 AccessDeniedException 使用 SpringBoot 上传文件报错 java.nio.file.AccessDeniedException: /xx/xx/xx&#xff0c;这个异常意味着程序在试图访问该目录…

SDN — Google B4 SDN WAN 网络架构

目录 文章目录 目录Google B4 SDN WAN 网络B4 网络架构物理设备层局部网络控制层全局控制层Hybrid SDN 模式Google B4 SDN WAN 网络 Google 的 WAN 有 2 张网络(Two Backbones): B2(I-Scale Network):数据中心互联 Internet(POP)的网络,用于面向 Internet 用户访问,…

MQTT(1):MQTT协议介绍

随着 5G 时代的来临&#xff0c;万物物联的伟大构想正在成为现实。联网的物联网设备在 2018 年已经达到了 70 亿&#xff0c;在未来两年&#xff0c;仅智能水电气表就将超过10亿 海量的设备接入和设备管理对网络带宽、通信协议以及平台服务架构都带来了很大挑战。对于物联网协议…

「OceanBase 4.1 体验」|docker-compose快速部署OceanBase数据库——筑梦之路

OceanBase数据库简介 官方网站&#xff1a;https://www.oceanbase.com/softwarecenter 大名鼎鼎的OceanBase数据库&#xff0c;在多个双十一购物节上历经验证&#xff0c;今天就来体验一下当前最新版本 4.1。 OceanBase 4.1 版本技术文档&#xff1a;https://www.oceanbase.c…

【JAVAEE】认识网络及网络通信

目录 1.网络发展史 1.1独立模式 1.2网络互连 1.2.1局域网 1.2.2广域网 2.网络通信基础 2.1IP地址 2.2端口号 2.3协议 2.4五元组 2.5协议分层 2.5.1什么是协议分层 2.5.2协议分层的作用 2.5.3TCP/IP五层&#xff08;或四层&#xff09;模型 3.封装和分用 1.网络发…

关于CSDN如何获得铁粉

一、发表高质量技术博客 获得铁粉首先是需要有粉丝关注&#xff0c;在CSDN有粉丝关注&#xff0c;就需要多发表写技术文章而且最好是高质量文章&#xff0c;条理清晰&#xff0c;复合当下主流技术&#xff0c;或者新的技术方向&#xff0c;图文并茂的那种。这样通过搜索引擎搜到…

数据库 查询执行(1) 多路归并

用户输入一个sql语句 sql解析器 将sql语句转换成一个 关系代数 但是同时也会附加一些 操作 (下方未体现)。 外存排序 原理 外部数据元素太多无法一次性的读入内存 通常设计者 会让磁盘块的大小和缓冲区的大小相等 。 首先构造初始的归并段 分别从横向和纵向 体会这…

Tcl-7. binary命令

binary format 就是将数值根据规定模式对 Tcl 的普通数据进行二进制压缩&#xff0c; 而 binary scan 作用相反&#xff0c;是从二进制数值恢复 Tcl 普通数据。 下面列举了这两个 binary 命令的作用。 首先用 binary format 对数值字符串”25664”进行二进制压缩&#xff0c;…