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

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

背景:

  1. 系统架构:SpringCloud分布式
  2. 持久层:MyBatis
  3. 前端:前后分离vue.js/bootstrap等.

后台提供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接口界面如下:

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+缓存分页

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

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

  1. 先把所有数据记录查询出来
  2. 数据库实体再次封装为查询实体
  3. 纯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+;在没有条件查询数据库的时候,耗时久

 

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

 

 

同名原创公众号: 程序大视界

 

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
本文介绍了一个基于Spring Boot、Spring Cloud和Vue前后端分离的项目实战。这个项目是一个简单的在线商城,包含了用户注册、登录、商品展示、购物车、订单管理等功能。通过这个项目,读者可以深入理解前后端分离的架构模式和互联网应用的开发方式。 首先,文章介绍了前后端分离的基本概念和优势。前后端分离是将应用的前端和后端代码分开来开发,使得前端和后端具有独立的开发周期和技术栈,进而提高了开发效率和代码质量。同时,前后端分离还可以提供更好的用户体验和灵活性,对于互联网应用来说尤为重要。 接下来,文章介绍了项目的架构和技术栈。项目采用了Spring Boot和Spring Cloud框架来实现后端代码,采用MyBatis作为ORM框架和Redis作为缓存中间件。同时,项目还采用了Vue.js作为前端框架和Element UI组件库来实现前端页面。通过这些开源框架和组件,可以快速搭建一个前后端分离的互联网应用。 然后,文章介绍了项目的核心功能和代码实现。在用户注册和登录方面,项目采用了Spring Security框架和JWT令牌来实现用户认证和授权,保证了用户信息的安全性。在商品展示和购物车方面,项目采用了Vue.js来实现前端页面和事件处理。在订单管理方面,项目采用了MyBatis Plus来实现订单数据的持久化和分页查询。 最后,文章介绍了项目的测试和优化。通过对项目的压力测试和性能测试,文章发现项目还存在一些性能瓶颈和安全隐患,可以通过优化数据库查询、缓存配置和代码实现来提高应用的性能和安全性。 总之,这篇文章介绍了一个基于Spring Boot、Spring Cloud和Vue前后端分离的项目实战,通过实现一个在线商城的功能,展示了前后端分离的开发模式和互联网应用的开发技术栈。本文可以作为前后端分离开发的入门教程,也可以作为互联网应用开发的参考文档。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

程序大视界

原创不易,请给点支持和鼓励吧

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值