1. 什么是深分页?
比如你有一个表,里面有 1000 万条数据,你想看第 10 万页,每页 10 条数据。
SQL 大概是这样:
sql
SELECT * FROM user ORDER BY id LIMIT 1000000, 10;
这里的 1000000 就是偏移量(OFFSET),表示跳过前 100 万条,再取 10 条。
这种写法在数据量很大、页码很深的时候会特别慢,因为数据库必须先把前 100 万条数据找出来,再扔掉,最后只返回 10 条。即使有索引,它也要做很多无用功。
2. 怎么优化?
优化的核心思想就一句话:尽量让数据库不要跳过那么多数据。
有两种最常用的基础方法:
方法一:记住上一页的最后一条数据,直接取下一页(游标)
假设你刚才看了第 1 页,最后一条记录的 id 是 100。那么看第 2 页的时候,不要用 OFFSET,而是直接取 id > 100 的 10 条:
sql
SELECT * FROM user
WHERE id > 100
ORDER BY id
LIMIT 10;
这里的100可以直接从前端传过来,这种方法叫游标
这样数据库直接从 id=101 开始往后取 10 条,不用跳,非常快。
缺点:这种写法只适合"上一页、下一页"这种翻页方式,不能直接跳到第 100 页。但大多数业务场景下,用户其实很少真的去点第 100 页,所以这种方法是性价比最高的。
方法二:先只查主键,再根据主键取完整数据
如果你必须支持跳页(比如用户可以直接输入页码),那就没办法避免偏移量。但我们可以让数据库在扫描的时候只扫描主键(或索引),少做点无用功。
原来的慢 SQL(先查所有字段,再跳 100 万行):
sql
SELECT * FROM user ORDER BY id LIMIT 1000000, 10;
优化后的 SQL(分两步,但写成一个 SQL):
sql
SELECT * FROM user
WHERE id IN (
SELECT id FROM user
ORDER BY id
LIMIT 1000000, 10
);
或者写成 JOIN 形式(更通用):
sql
SELECT u.*
FROM user u
JOIN (
SELECT id FROM user
ORDER BY id
LIMIT 1000000, 10
) tmp ON u.id = tmp.id;
为什么这样快?
子查询 SELECT id FROM user ... 只查 id 列,如果 id 是主键,数据库可以只扫描索引,不用把整行数据都读出来,速度会快很多。然后外层再用这些 id 去取完整的行,因为 id 是主键,回表也是很快的。
3. 总结
- 如果你能用"记住上一页的最后一条数据"的方法,就用它,这是最快最稳的。
- 如果必须支持跳页,就用"先查主键,再回表"的方法。
- 无论用哪种方法,都要保证
ORDER BY的字段上有索引。
再举个例子:如果你的分页是按时间排序的,比如 ORDER BY create_time DESC,那么方法一就变成:
sql
-- 上一页最后一条的时间是 '2024-01-01 10:00:00'
SELECT * FROM user
WHERE create_time < '2024-01-01 10:00:00'
ORDER BY create_time DESC
LIMIT 10;
方法二就改成:
sql
SELECT u.*
FROM user u
JOIN (
SELECT id FROM user
ORDER BY create_time DESC
LIMIT 1000000, 10
) tmp ON u.id = tmp.id
ORDER BY u.create_time DESC;