记录一个分页插件问题

在开发的时候遇到一个问题,同样的SQL条件,两次查询结果不一致。

门店查询获取到76条数据,总数显示为0。

商家查询获取到10条数据,总是显示为76.

不同身份进入到接口,通过进行权限的限制set值,所获得的查询前置条件是一样的。如下

image-20240124120508870

通过日志打印拿到两次查询的SQL

如下:

-- 开发测试商家会员列表
SELECT
    c.id, c.nickname, c.birthday, c.avatar, ca.grade, c.gender, m.agent_id , ca.consumer_id, ca.merchant_id, ca.account_balance, ca.account_constant, ca.account_credits, ca.remark, ca.integral, ca.remaining_seconds, ca.balance, ca.label, ca.amount_of_gift
FROM consumer AS c
    LEFT JOIN consumer_account AS ca ON c.id = ca.consumer_id
    LEFT JOIN merchant AS m ON m.id = ca.merchant_id
WHERE ca.merchant_id = 2 GROUP BY ca.id order by ca.id LIMIT 10;

-- 开发测试门店会员列表
SELECT c.id, c.nickname, c.birthday, c.avatar, ca.grade, c.gender, m.agent_id , ca.consumer_id, ca.merchant_id, ca.account_balance, ca.account_constant, ca.account_credits, ca.remark, ca.integral, ca.remaining_seconds, ca.balance, ca.label, ca.amount_of_gift
FROM consumer AS c
    LEFT JOIN consumer_account AS ca ON c.id = ca.consumer_id
    LEFT JOIN merchant AS m ON m.id = ca.merchant_id
WHERE ca.merchant_id = 2 GROUP BY ca.id order by ca.id

很明显发现分页查询条件没有了,

看到代码里面是通过分页插件去执行分页的

image-20240124140145918

然后在 开发测试门店会员列表的SQL去page.getTotal()得到值为0

因为分页插件的获取原理是

total:SELECT * FROM (查询SQL)

row:查询SQL拼接LIMIT语句

通过断点debug发现在构建分页的时候startPage的size都是0

image-20240124141142660

然后门店执行到SQL执行语句的时候变成了1

image-20240124141255812

很明显,可以观察到门店与商家的差距是因为门店有额外去查询商家id作为查询条件这一操作,商家查询是直接从token里面拿的商家id。

image-20240124141737533

将构建分页代码后移到查询语句处,优化代码查询得出结果,一致! 问题解决。

image-20240124141941368

查询得出结果,一致! 问题解决。

image-20240124142147046

Spring Boot是一个用于快速开发Java应用程序的框架,它基于Spring框架,并提供了自动配置和约定优于配置的原则。分页插件是在Spring Boot中实现分页功能的一种方式,可以方便地对数据库查询结果进行分页处理。 要手写一个分页插件,你可以按照以下步骤进行: 1. 创建一个Page类,用于封装分页相关的信息,如当前页码、每页显示数量、总记录数等。 2. 在数据库查询方法中,根据传入的Page对象,使用limit语句限制查询结果的范围,实现分页功能。 3. 在业务层或控制层中,根据用户请求的页码和每页显示数量,创建Page对象,并将其传递给数据库查询方法。 4. 在前端页面中,显示分页导航栏和查询结果列表,并提供翻页功能。 下面是一个简单的示例代码,演示了如何手写一个基于MySQL数据库分页插件: ```java // Page类 public class Page { private int pageNum; // 当前页码 private int pageSize; // 每页显示数量 private int total; // 总记录数 // 省略getter和setter方法 } // 数据库查询方法 public List<User> getUsers(Page page) { int offset = (page.getPageNum() - 1) * page.getPageSize(); String sql = "SELECT * FROM user LIMIT ?, ?"; return jdbcTemplate.query(sql, new Object[]{offset, page.getPageSize()}, new BeanPropertyRowMapper<>(User.class)); } // 业务层或控制层 public List<User> getUsers(int pageNum, int pageSize) { Page page = new Page(); page.setPageNum(pageNum); page.setPageSize(pageSize); page.setTotal(userDao.countUsers()); // 获取总记录数 return userService.getUsers(page); } // 前端页面 <!-- 分页导航栏 --> <div class="pagination"> <a href="?pageNum=1">首页</a> <a href="?pageNum=${page.pageNum-1}">上一页</a> <a href="?pageNum=${page.pageNum+1}">下一页</a> <a href="?pageNum=${page.totalPage}">尾页</a> </div> <!-- 查询结果列表 --> <table> <thead> <tr> <th>姓名</th> <th>年龄</th> </tr> </thead> <tbody> <c:forEach items="${users}" var="user"> <tr> <td>${user.name}</td> <td>${user.age}</td> </tr> </c:forEach> </tbody> </table> ``` 希望以上示例能帮助到你!如果有任何问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值