SpringCloud+MyBatis分页处理(前后端分离)

简介: 分页处理,这是做JavaWeb项目中常见的场景。

分页处理,这是做JavaWeb项目中常见的场景。


背景


  1. 系统架构:SpringCloud分布式
  2. 持久层:MyBatis
  3. 前端:前后分离vue.js/bootstrap等.
  4. 后台提供restful api 接口,前端访问后端接口展示数据。


2种方式提供分页处理方案:


一、直接MyBatis数据库进行分页


controller接口

@ApiImplicitParams({            @ApiImplicitParam(name = "categoryId", value = "支付渠道大类Id", required = false, dataType = "Long",  paramType = "query"),            @ApiImplicitParam(name = "payChannelId", value = "支付渠道ID", required = false, dataType = "Long", paramType = "query"),            @ApiImplicitParam(name = "bankCode", value = "银行名称code", required = false, dataType = "String", paramType = "query"),            @ApiImplicitParam(name = "startTime", value = "创建时间起始(示例:2018-09-05)", required = false, dataType = "String", paramType = "query", length = 10),            @ApiImplicitParam(name = "endTime", value = "创建时间截止(示例:2018-09-05)", required = false, dataType = "String", paramType = "query", length = 10),            @ApiImplicitParam(name = "pageNum", value = "查询开始页", required = true, dataType = "int", paramType = "query"),            @ApiImplicitParam(name = "pageSize", value = "查询的页面大小不需要分页则把此值填大一点", required = true, dataType = "int", paramType = "query")})    public PageResult<ChannelBaseDataResp> queryPayChannel(                                            @RequestParam(value = "categoryId", required = false) Long categoryId,                                            @RequestParam(value = "payChannelId", required = false) Long payChannelId,                                            @RequestParam(value = "bankCode", required = false) String bankCode,                                            @RequestParam(value = "startTime", required = false) String startTime,                                            @RequestParam(value = "endTime", required = false) String endTime,                                            @RequestParam(value = "pageNum", required = true) Integer pageNum,                                            @RequestParam(value = "pageSize", required = true) Integer pageSize) {        PageResult<ChannelBaseDataResp> response = new PageResult<ChannelBaseDataResp>();        try {            int endRowNo = pageNum * pageSize;            int beginRowNo = (pageNum - 1) * pageSize + 1;            return channelBaseDataService.queryBaseData(categoryId,payChannelId,bankCode,startTime,endTime,beginRowNo,endRowNo);        }  catch (Exception e) {            response = new PageResult<ChannelBaseDataResp>();            if (e instanceof AppException) {
                response.setCode(((AppException) e).getErrorCode());                response.setResult(((AppException) e).getErrorMsg());            } else {                response.setCode(CumReturnCode.SYSTEM_EXCEPTION.code);                response.setResult(CumReturnCode.SYSTEM_EXCEPTION.message);            }            return response;        }    }

Swagger-ui接口界面如下:

16182cd7b4ce77e714f7878e71421a0e_640_wx_fmt=png&wxfrom=5&wx_lazy=1&wx_co=1.png


pageNum 和 pageSize 用于前端分页的参数,pageNum:表示页码第几页,pageSize:表示每页展示数据数量。


所有查询条件参数,在mapper-xml里进行处理,再利用数据库oracle本身的rownum行值进行分页。


xml代码:


<select id="queryBaseDataList" resultMap="BaseResultMap">    select    ID, PAY_CHANNEL_MAIN_ID, PAY_CHANNEL_ID,    BANK_CODE, CREATE_TIME, UPDATE_TIME,UPDATOR    from (    select A.*, rownum RN    from (    select ID, PAY_CHANNEL_MAIN_ID, PAY_CHANNEL_ID,    BANK_CODE, CREATE_TIME, UPDATE_TIME,UPDATOR    from CUM_PAY_CHANNEL_BASE    <where>      <if test="beginCreateTime != null">        CREATE_TIME <![CDATA[ >= ]]>  #{beginCreateTime,jdbcType=TIMESTAMP}      </if>      <if test="endCreateTime != null">        AND CREATE_TIME <![CDATA[ <= ]]>  #{endCreateTime,jdbcType=TIMESTAMP}      </if>      <if test="categoryId != null and categoryId != '' ">        AND PAY_CHANNEL_MAIN_ID = #{categoryId,jdbcType=DECIMAL}      </if>      <if test="payChannelId != null and payChannelId != ''">        AND PAY_CHANNEL_ID = #{payChannelId,jdbcType=DECIMAL}      </if>      <if test="bankCode != null and bankCode != ''">        AND BANK_CODE = #{bankCode,jdbcType=VARCHAR}      </if>    </where>    order by CREATE_TIME desc    ) A    where rownum &lt;= #{endRowNo,jdbcType=DECIMAL}    )    where RN &gt;= #{beginRowNo,jdbcType=DECIMAL}  </select>

优点:直观、方便、易排查问题。 缺点:访问数据库过于频繁,未利用到mybatis本身的缓存优势。


二、Java+缓存分页



这种方法对于前端而言是没变化,无感的。


只是在后端处理稍作变动,思路:

  • 先把所有数据记录查询出来
  • 数据库实体再次封装为查询实体
  • 纯java代码进行分页
  • controller接口不变
  • service业务处理——>数据库crud操作变动
  • 把所有数据库记录查询处理


<select id="queryBaseDataList" resultMap="BaseResultMap">    select    ID, PAY_CHANNEL_MAIN_ID, PAY_CHANNEL_ID,    BANK_CODE, CREATE_TIME, UPDATE_TIME,UPDATOR    from CUM_PAY_CHANNEL_BASE    <where>      <if test="beginCreateTime != null">        CREATE_TIME <![CDATA[ >= ]]>  #{beginCreateTime,jdbcType=TIMESTAMP}      </if>      <if test="endCreateTime != null">        AND CREATE_TIME <![CDATA[ <= ]]>  #{endCreateTime,jdbcType=TIMESTAMP}      </if>      <if test="categoryId != null and categoryId != '' ">        AND PAY_CHANNEL_MAIN_ID = #{categoryId,jdbcType=DECIMAL}      </if>      <if test="payChannelId != null and payChannelId != ''">        AND PAY_CHANNEL_ID = #{payChannelId,jdbcType=DECIMAL}      </if>      <if test="bankCode != null and bankCode != ''">        AND BANK_CODE = #{bankCode,jdbcType=VARCHAR}      </if>    </where>    order by CREATE_TIME desc</select>

再按照pageNum和pageSize进行分页处理


//以分页形式输出给前端        List<ChannelRouteGroupResp> resultList = new ArrayList<>();        if(groupRespList.size() >= endRowNo){            for(int i=(beginRowNo-1); i< endRowNo; i++){                resultList.add(groupRespList.get(i));            }        }else{            for(int i=(beginRowNo-1); i< groupRespList.size(); i++){                resultList.add(groupRespList.get(i));            }        }

优点:利用了mybatis缓存机制,分页查询快速,减少数据库访问次数。


缺点:当数据量大的时候,100W+;在没有条件查询数据库的时候,耗时久


前端分页后的界面展示如下:

da0a2e198fb7e846bb26b4a79a9e1bae_640_wx_fmt=png&wxfrom=5&wx_lazy=1&wx_co=1.png

相关文章
|
6月前
SpringBoot+Mybatis-Plus+PageHelper分页+多条件查询
SpringBoot+Mybatis-Plus+PageHelper分页+多条件查询
158 0
|
3月前
|
SQL XML Java
8、Mybatis-Plus 分页插件、自定义分页
这篇文章介绍了Mybatis-Plus的分页功能,包括如何配置分页插件、使用Mybatis-Plus提供的Page对象进行分页查询,以及如何在XML中自定义分页SQL。文章通过具体的代码示例和测试结果,展示了分页插件的使用和自定义分页的方法。
8、Mybatis-Plus 分页插件、自定义分页
|
3天前
|
SQL 缓存 Java
【详细实用のMyBatis教程】获取参数值和结果的各种情况、自定义映射、动态SQL、多级缓存、逆向工程、分页插件
本文详细介绍了MyBatis的各种常见用法MyBatis多级缓存、逆向工程、分页插件 包括获取参数值和结果的各种情况、自定义映射resultMap、动态SQL
【详细实用のMyBatis教程】获取参数值和结果的各种情况、自定义映射、动态SQL、多级缓存、逆向工程、分页插件
|
27天前
|
SQL JSON Java
mybatis使用三:springboot整合mybatis,使用PageHelper 进行分页操作,并整合swagger2。使用正规的开发模式:定义统一的数据返回格式和请求模块
这篇文章介绍了如何在Spring Boot项目中整合MyBatis和PageHelper进行分页操作,并且集成Swagger2来生成API文档,同时定义了统一的数据返回格式和请求模块。
44 1
mybatis使用三:springboot整合mybatis,使用PageHelper 进行分页操作,并整合swagger2。使用正规的开发模式:定义统一的数据返回格式和请求模块
|
2月前
|
SQL Java 数据库连接
解决mybatis-plus 拦截器不生效--分页插件不生效
本文介绍了在使用 Mybatis-Plus 进行分页查询时遇到的问题及解决方法。依赖包包括 `mybatis-plus-boot-starter`、`mybatis-plus-extension` 等,并给出了正确的分页配置和代码示例。当分页功能失效时,需将 Mybatis-Plus 版本改为 3.5.5 并正确配置拦截器。
492 6
解决mybatis-plus 拦截器不生效--分页插件不生效
|
2月前
|
SQL XML Java
springboot整合mybatis-plus及mybatis-plus分页插件的使用
这篇文章介绍了如何在Spring Boot项目中整合MyBatis-Plus及其分页插件,包括依赖引入、配置文件编写、SQL表创建、Mapper层、Service层、Controller层的创建,以及分页插件的使用和数据展示HTML页面的编写。
springboot整合mybatis-plus及mybatis-plus分页插件的使用
|
3月前
|
Java 数据库 Spring
MyBatisPlus分页插件在SpringBoot中的使用
这篇文章介绍了如何在Spring Boot项目中配置和使用MyBatis-Plus的分页插件,包括创建配置类以注册分页拦截器,编写测试类来演示如何进行分页查询,并展示了测试结果和数据库表结构。
MyBatisPlus分页插件在SpringBoot中的使用
|
3月前
|
SQL Java 关系型数据库
MyBatis-Plus 分页魅力绽放!紧跟技术热点,带你领略数据分页的高效与便捷
【8月更文挑战第29天】在 Java 开发中,数据处理至关重要,尤其在大量数据查询与展示时,分页功能尤为重要。MyBatis-Plus 作为一款强大的持久层框架,提供了便捷高效的分页解决方案。通过封装数据库分页查询语句,开发者能轻松实现分页功能。在实际应用中,只需创建 `Page` 对象并设置页码和每页条数,再通过 `QueryWrapper` 构建查询条件,调用 `selectPage` 方法即可完成分页查询。MyBatis-Plus 不仅生成分页 SQL 语句,还自动处理参数合法性检查,并支持条件查询和排序等功能,极大地提升了系统性能和稳定性。
52 0
|
4月前
|
SQL 关系型数据库 Java
mybatis-分页
1. MyBatis RowBounds分页:先查询所有结果,再进行内存分页。 2. PageHelper插件:自动识别数据库类型并添加对应分页关键字,分两步执行:添加分页查询,然后查询总数。 3. SQL分页:直接在SQL中使用`LIMIT`或`ROWNUM`等进行分页。 4. 数组分页:DAO层查询所有数据,Service层通过`subList`方法实现分页。 5. 拦截器分页:自定义拦截器对特定方法进行拦截,并在SQL语句中添加分页参数。 6. 总结:逻辑分页适合小数据量,物理分页适合大数据量避免内存溢出。物理分页优于逻辑分页。