mysql如何分页查询数据_mysql分页操作方法

7次阅读

mysql分页应使用limit (page-1)*size, size,注意offset从0开始且需校验参数;深分页性能差因逐行扫描,推荐游标分页(where+排序字段)替代。

mysql如何分页查询数据_mysql分页操作方法

MySQL 用 LIMIT 实现分页的正确写法

MySQL 分页靠 LIMIT,但不是简单加两个数字就完事。最常用的是 LIMIT offset, size 形式,比如查第 2 页、每页 10 条:LIMIT 10, 10(跳过前 10 条,取接下来 10 条)。

注意:第一个参数是偏移量(从 0 开始),不是页码。页码为 page 时,offset 应为 (page - 1) * size。直接写 LIMIT 2, 10 想查第 2 页?那是错的——它跳过前 2 条,不是前 2 页。

  • 后端传参建议校验 page > 0size 在合理范围(如 1–100),避免 LIMIT -5, 10 或超大 offset 导致 SQL 报错或性能崩塌
  • order by 必须显式指定,否则分页结果可能乱序——MySQL 不保证无排序时的行顺序
  • 不要用 LIMIT 1000000, 20 查深分页,MySQL 仍要扫描前 1000020 行,响应会明显变慢

深分页性能差,为什么 OFFSET 越大越慢

LIMIT offset, size 的底层逻辑是:MySQL 先定位到第 offset + 1 行,这需要逐行遍历前面所有记录(即使不返回)。当 offset 达到几十万,IO 和 CPU 开销陡增,查询可能从毫秒级变成秒级甚至超时。

典型现象:select * FROM orders ORDER BY id DESC LIMIT 999990, 10 执行缓慢,而 LIMIT 0, 10 飞快。

  • 根本原因:没有利用索引跳过前 N 行,而是“数着走”
  • 优化方向不是调大 buffer,而是改用游标分页(cursor-based pagination)
  • 如果必须用 offset,确保 ORDER BY 字段有索引,且类型匹配(比如 idBIGINT,就别拿字符串去比较)

用 WHERE + 排序字段替代 LIMIT OFFSET 做游标分页

游标分页不依赖行号,而是记住上一页最后一条的排序值,下一页查“比它更小/更大的记录”。适用于按主键或时间戳分页,且数据写入相对稳定(无频繁插入打乱顺序)。

例如按 id 降序分页,第一页取最大 10 条:SELECT * FROM users ORDER BY id DESC LIMIT 10;假设最后一条 id = 12345,第二页就查:SELECT * FROM users WHERE id 。

  • WHERE 条件必须和 ORDER BY 字段一致,且能命中索引(EXPLAINtype 是否为 range
  • 如果排序字段不唯一(比如多个记录 created_at 相同),需追加唯一字段(如 id)做二级排序,避免漏数据或重复
  • 前端不能跳页(比如直接输“第 100 页”),只能“下一页/上一页”,因为没存中间页的游标值

MySQL 8.0+ 的 ROW_NUMBER() 能否简化分页逻辑

可以,但不推荐用于常规分页。ROW_NUMBER() 是窗口函数,适合做“取排名前 N”或“按组分页”,但对全表分页并无性能优势——它仍需计算全部行的序号,再过滤,本质还是扫全表。

示例:SELECT * FROM (SELECT *, ROW_NUMBER() OVER (ORDER BY id) AS rn FROM users) t WHERE rn BETWEEN 11 AND 20。看似直观,实际执行计划里常出现 using temporary; Using filesort,比带索引的 WHERE 游标还慢。

  • ROW_NUMBER() 更适合一次性导出、报表类场景,而非在线 API 分页
  • 若真要用,务必在 OVER() 中的 ORDER BY 字段建索引,否则性能灾难
  • 注意:MySQL 5.7 及更早版本不支持窗口函数,强行用会报错 function xxx.ROW_NUMBER does not exist

游标分页的健壮性远高于 offset 分页,但要求业务接受“不可跳页”和“游标值需透传”的约束。OFFSET 看似简单,却是线上慢查询的高频诱因。

text=ZqhQzanResources