MySQL 深分页越来越慢?从 LIMIT OFFSET 改成游标分页

分页查询是后台系统中最常见的功能之一。

数据量较小时,通常会直接使用:

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,而是根据业务交互方式选择合适的分页模型。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值