分页查询是后台系统中最常见的功能之一。
数据量较小时,通常会直接使用:
SELECT *
FROM meeting_record
ORDER BY id DESC
LIMIT 20 OFFSET 0;
翻到第 1000 页后,SQL 可能变成:
SELECT *
FROM meeting_record
ORDER BY id DESC
LIMIT 20 OFFSET 19980;
随着页码不断增加,查询速度往往会越来越慢。这类问题通常被称为"深分页"。
一、为什么 OFFSET 越大越慢?
LIMIT 20 OFFSET 19980 并不是让数据库直接从第 19981 条数据开始读取。
数据库通常需要:
先找到前 20000 条数据
丢弃前 19980 条
最后返回 20 条
当 OFFSET 达到几十万甚至几百万时,大量扫描结果最终都会被丢弃。
例如:
SELECT *
FROM meeting_record
ORDER BY id DESC
LIMIT 20 OFFSET 500000;
为了返回 20 条记录,数据库可能需要扫描并跳过前面的 50 万条数据。
如果查询还包含排序、回表或复杂过滤,性能下降会更加明显。
二、先确认索引是否有效
假设查询条件为:
SELECT *
FROM meeting_record
WHERE user_id = 10001
ORDER BY id DESC
LIMIT 20 OFFSET 100000;
可以增加联合索引:
CREATE INDEX idx_user_id
ON meeting_record(user_id, id);
然后通过 EXPLAIN 查看执行计划:
EXPLAIN
SELECT *
FROM meeting_record
WHERE user_id = 10001
ORDER BY id DESC
LIMIT 20 OFFSET 100000;
重点关注:
key:是否使用预期索引
rows:预计扫描多少行
Extra:是否出现 filesort
索引可以减少扫描范围,但无法完全消除大 OFFSET 带来的开销。
三、使用游标分页
如果业务不要求用户直接跳转到任意页,可以改成游标分页。
第一页:
SELECT *
FROM meeting_record
WHERE user_id = 10001
ORDER BY id DESC
LIMIT 20;
假设最后一条记录的 ID 是 9520,下一页查询:
SELECT *
FROM meeting_record
WHERE user_id = 10001
AND id < 9520
ORDER BY id DESC
LIMIT 20;
前端只需要把上一页最后一条记录的 ID 传回来:
{
"lastId": 9520,
"pageSize": 20
}
这种方式不需要数据库跳过大量无用记录,查询性能通常更加稳定。
Java 示例:
public List<MeetingRecord> queryNextPage(
Long userId,
Long lastId,
Integer pageSize
) {
return meetingRecordMapper.queryNextPage(
userId,
lastId,
pageSize
);
}
对应 SQL:
SELECT *
FROM meeting_record
WHERE user_id = #{userId}
AND id < #{lastId}
ORDER BY id DESC
LIMIT #{pageSize};
四、游标分页有什么限制?
游标分页也不是万能方案。
它不适合:
直接跳转到第 100 页
展示准确的总页数
数据排序字段频繁变化
它更适合:
下拉加载
消息列表
操作日志
会议记录
订单流水
时间线
例如**同言翻译(Transync AI)**在生成实时翻译记录和 AI 会议总结后,如果用户需要浏览大量历史会议,记录列表就更适合使用游标分页,而不是不断增大的 OFFSET。
五、排序字段不唯一怎么办?
如果按创建时间排序:
ORDER BY created_at DESC
可能有多条数据拥有相同时间。
只使用:
WHERE created_at < ?
可能导致部分记录重复或遗漏。
更稳妥的方式是使用复合游标:
SELECT *
FROM meeting_record
WHERE user_id = 10001
AND (
created_at < '2026-06-16 10:00:00'
OR (
created_at = '2026-06-16 10:00:00'
AND id < 9520
)
)
ORDER BY created_at DESC, id DESC
LIMIT 20;
同时建立索引:
CREATE INDEX idx_user_time_id
ON meeting_record(user_id, created_at, id);
这样即使创建时间相同,也可以通过 ID 保证稳定顺序。
六、必须跳页时怎么优化?
部分后台管理系统确实需要跳转页码。
可以先使用覆盖索引查出主键:
SELECT id
FROM meeting_record
WHERE user_id = 10001
ORDER BY id DESC
LIMIT 20 OFFSET 100000;
再根据主键查询完整数据:
SELECT *
FROM meeting_record
WHERE id IN (...);
也可以写成关联查询:
SELECT m.*
FROM meeting_record m
JOIN (
SELECT id
FROM meeting_record
WHERE user_id = 10001
ORDER BY id DESC
LIMIT 20 OFFSET 100000
) t ON m.id = t.id
ORDER BY m.id DESC;
这种方式仍然需要扫描较大的 OFFSET,但可以减少扫描阶段的回表数据量。
七、分页优化检查清单
遇到深分页问题时,可以依次检查:
1. 查询条件是否建立合适索引
2. ORDER BY 是否可以使用索引
3. 是否真的需要跳转到任意页
4. 是否可以改成游标分页
5. 排序字段是否唯一
6. 是否需要使用复合游标
7. 是否可以先查主键再查询完整数据
8. 是否需要限制最大可翻页数
总结
MySQL 深分页变慢的主要原因,是数据库需要扫描并丢弃大量数据。
如果业务是消息流、日志、会议记录或订单流水,优先考虑:
索引排序 + 游标分页
如果必须支持任意页跳转,可以尝试:
覆盖索引 + 主键回表
分页优化的关键不是单纯修改 LIMIT,而是根据业务交互方式选择合适的分页模型。

