news 2026/4/23 12:51:53

Java分页查询方式总结

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Java分页查询方式总结

文章目录

    • 分页查询核心思路
    • 常见分页实现方式
      • MyBatis
      • MyBatis-Plus

分页查询核心思路

分页的本质是限制查询结果的条数+跳过指定行数,并查询总记录数(用于计算总页数)。核心参数:

  • pageNum:当前页码(从 1 开始)
  • pageSize:每页显示条数
  • 起始行计算:startRow = (pageNum - 1) * pageSize
  • 总页数计算:totalPages = (totalCount + pageSize - 1) / pageSize(向上取整)

常见分页实现方式

MyBatis

1、mapper手动拼SQL

Mapper 接口:

import org.apache.ibatis.annotations.Param; import java.util.List; public interface UserMapper { // 查询分页数据 List<User> selectUserByPage(@Param("startRow") int startRow, @Param("pageSize") int pageSize); // 查询总条数 int selectUserTotalCount(); }

Mapper.xml:

<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd"> <mapper namespace="com.example.mapper.UserMapper"> <!-- 分页查询数据 --> <select id="selectUserByPage" resultType="com.example.entity.User"> SELECT id, name, age FROM user LIMIT #{startRow}, #{pageSize} </select> <!-- 查询总条数 --> <select id="selectUserTotalCount" resultType="int"> SELECT COUNT(*) FROM user </select> </mapper>

2、PageHelper 插件

需要在 pom.xml 中添加 PageHelper 依赖

<dependency> <groupId>com.github.pagehelper</groupId> <artifactId>pagehelper-spring-boot-starter</artifactId> <version>1.4.7</version> </dependency>

在调用时添加如下代码

// 开启分页(只对紧接着的第一个查询生效) PageHelper.startPage(pageNum, pageSize);

MyBatis-Plus

1、使用内置分页插件 PaginationInnerInterceptor

Page<Forlan>page=newPage<>(pageNum,pageSize);QueryWrapper<Forlan>queryWrapper=newQueryWrapper<>();queryWrapper.eq("id",id);mapper.selectPage(page,queryWrapper);service.lambdaQuery().page(newPage<>(pageNum,pageSize))

2、基于偏移量的分页(升级写法,自己控制结束)

核心参数:

  1. LIMIT N
    表示“只返回最多 N 条记录”。
    通常用来控制每页显示的数据条数。
  2. OFFSET M
    表示“跳过前 M 条记录”,从第 M+1 条开始取数据。
    用于定位到当前请求的页码起始位置。
    核心:service.query().last("LIMIT " + batchSize + " OFFSET " + offset).list();
finalintbatchSize=1000;List<Forlan>allResults=newArrayList<>();intoffset=0;booleanhasMoreResults=true;while(hasMoreResults){try{List<Forlan>batchResults=service.query().last("LIMIT "+batchSize+" OFFSET "+offset).list();if(batchResults==null||batchResults.isEmpty()){hasMoreResults=false;}else{allResults.addAll(batchResults);offset+=batchResults.size();}}catch(Exceptione){log.info("Error querying batch: ",e);hasMoreResults=false;}}returnallResults;

上面的写法,存在问题:随着 offset 增大,性能下降严重,对大数据量场景不友好,适合深度翻页,比如:

-- 查询第10001页,每页10条数据 SELECT * FROM products ORDER BY id LIMIT 10 OFFSET 100000;

这条SQL的执行逻辑并非直接定位到第100,001条记录。MySQL的实际处理过程是:从存储引擎中读取满足条件的前 100010 (OFFSET + LIMIT) 条记录,在服务层(Server Layer)对这些记录进行排序,抛弃前面的 100000 条记录,返回最终的 10 条记录。

所以,OFFSET 值越大,MySQL需要扫描、加载并最终抛弃的行数就越多,这导致了巨大的I/O和CPU资源浪费,是性能下降的直接原因。

1)延迟关联:优化后的写法

核心思想:先通过覆盖索引快速定位到目标页的主键ID,然后再关联原表获取完整的行数据,从而减少对主表数据的扫描。

LonglastId=0L;finalintbatchSize=1000;List<Forlan>allResults=newArrayList<>();booleanhasMoreResults=true;while(hasMoreResults){try{List<Forlan>batchResults=service.query().gt("id",lastId).last("LIMIT "+batchSize).list();if(batchResults==null||batchResults.isEmpty()){hasMoreResults=false;}else{allResults.addAll(batchResults);lastId=batchResults.get(batchResults.size()-1).getId();}}catch(Exceptione){log.info("Error querying batch: ",e);hasMoreResults=false;}}returnallResults;

存在问题,如果扫描的最小id在几千万,这时候首次查询也是非常耗费时间的,进一步优化的写法如下:

finalintbatchSize=1000;List<Forlan>allResults=newArrayList<>();booleanhasMoreResults=true;Forlanforlan=service.query().select("min(id) id").one();if(forlan==null){returnallResults;}LonglastId=forlan.getId()-1;while(hasMoreResults){try{List<Forlan>batchResults=service.query().gt("id",lastId).last("LIMIT "+batchSize).list();if(batchResults==null||batchResults.isEmpty()){hasMoreResults=false;}else{allResults.addAll(batchResults);lastId=batchResults.get(batchResults.size()-1).getId();}}catch(Exceptione){log.info("Error querying batch: ",e);hasMoreResults=false;}}returnallResults;

2)书签法

是目前性能最优的方案。它摒弃了OFFSET,通过上一页最后一条记录的唯一键值来定位下一页的起始位置,但要求主键或查询条件连续

假设我们按自增id排序,上一页返回的最后一条记录id为100000。不使用OFFSET,而是利用上一页的id进行定位

SELECT * FROM products WHERE id > 100000 ORDER BY id ASC LIMIT 10;

优点:查询性能恒定,不受分页深度影响,速度极快。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/14 5:55:39

360度VR全景设备技术测评与行业应用分析

1、三维空间数据采集的技术分野当前&#xff0c;360度VR全景设备已分化为多条清晰的技术路径&#xff0c;其差异远不止于分辨率高低&#xff0c;更在于核心工作原理、最终数据形态及适用的工程化场景。从基于AI的视觉重建&#xff0c;到依赖主动测距的精密扫描&#xff0c;不同…

作者头像 李华
网站建设 2026/4/23 10:49:51

偷懒也高效:帮你准备好的提示词复制范本(附场景)

很多人嘴上说着要“好好用 AI 提升效率”&#xff0c; 实际操作却是这样的&#xff1a;每次打开对话框&#xff0c;现场现编问题&#xff1b;想说清楚需求&#xff0c;结果越解释越乱&#xff1b;问到第三轮&#xff0c;自己都忘了最初想要什么。最后得出一个结论&#xff1a; …

作者头像 李华
网站建设 2026/4/23 10:45:21

ServiceNow预测阿联酋将在2030年新增超百万AI驱动岗位

ServiceNow预测&#xff0c;随着人工智能和数字技术在经济各领域的深度融合&#xff0c;阿联酋到2030年将新增超过103万个就业岗位&#xff0c;这凸显了该国将自身定位为全球AI中心的宏大雄心。这一预测正值公私部门持续投资AI驱动转型之际&#xff0c;相关举措包括《阿联酋203…

作者头像 李华
网站建设 2026/4/23 12:12:56

杂记 - 状态模式 VS. 责任链模式

目录 一、总体对比二、状态模式三、责任链模式四、扩展&#xff1a;手撸Java WebFilter实现 一、总体对比 状态模式和责任链模式都是行为型设计模式&#xff0c;但它们的意图和应用场景不同&#xff1a; 对比项状态模式责任链模式意图允许对象在内部状态改变时改变它的行为&a…

作者头像 李华
网站建设 2026/4/9 19:17:30

playwright工具(一)自动打开浏览器

playwright1、介绍Playwright 是一个由 Microsoft 开源的 端到端&#xff08;E2E&#xff09;自动化测试工具&#xff0c;主要用于测试 Web 应用。2、作用自动化测试 Chromium / Firefox / WebKit支持 多语言JavaScript / TypeScriptJavaPythonC#可用于&#xff1a;UI 自动化测…

作者头像 李华