深分页与游标

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;
相关推荐
六月雨滴3 分钟前
Oracle RMAN 安全与加密
安全·oracle·dba
曹牧14 分钟前
Oracle:UNIX时间戳
数据库·oracle·unix
XiaoLin laile19 分钟前
【无标题】
网络·数据库·人工智能
朝阳58132 分钟前
MySQL 主从复制 — Docker 双机灾备方案
数据库·mysql·docker
染翰33 分钟前
生产级 MySQL 内存占用过高排查指南
数据库·mysql
一 乐44 分钟前
网上订餐系统|基于springboot的网上订餐系统设计与实现(源码+数据库+文档)
java·数据库·spring boot·后端·论文·毕设·网上订餐系统
guslegend1 小时前
第3节:智能体配置表设计
数据库·人工智能
雷工笔记1 小时前
SQL系列2:PostgreSQL 日期时间字段类型选择指南
数据库·sql·postgresql
SAP上海工博云署1 小时前
2026年中小企业SAP服务商选型技术解析
大数据·运维·数据库·人工智能·信息可视化·运维开发·信息与通信
RestCloud1 小时前
版本迭代丨谷云科技ETLCloud V4.2版本更新速览
数据库·doris·etl·etlcloud·数据集成平台·datahub·ftp处理