当前位置:首页 > 文章列表 > 数据库 > MySQL > MySQL ORDER BY 排序变慢怎么办:从 filesort 到联合索引一步步排查

MySQL ORDER BY 排序变慢怎么办:从 filesort 到联合索引一步步排查

来源:17golang原创 2026-06-15 12:28:04 0浏览 收藏

我们先看一个列表接口的现场:订单列表平时几十毫秒返回,活动期间数据量上来后突然变成 1 秒多。业务同学说“只是按创建时间倒序查 20 条”,听起来不复杂,但数据库却越来越慢。

遇到这种问题,不要先急着加缓存。我们先把 SQL、筛选条件、排序字段和执行计划拿出来,看看到底是不是 ORDER BY 没吃到合适的索引,最后走到了额外排序。

适合人群

本文适合正在维护 MySQL 列表查询、分页接口、后台管理搜索页的开发者。你需要了解基本的索引、EXPLAINWHEREORDER BY

目录

  • 问题现场:列表接口排序突然变慢
  • 初步判断:先确认慢在排序还是筛选
  • 动手验证:用 EXPLAIN 看 Extra
  • 定位原因:索引顺序和排序方向没对齐
  • 修复方案:按筛选和排序设计联合索引
  • 验证结果:rows、Extra 和耗时一起看
  • 常见坑位和总结

问题现场:列表接口排序突然变慢

假设订单表结构简化如下:

CREATE TABLE orders (
  id BIGINT PRIMARY KEY,
  user_id BIGINT NOT NULL,
  status TINYINT NOT NULL,
  created_at DATETIME NOT NULL,
  amount DECIMAL(10, 2) NOT NULL
);

接口 SQL 是:

SELECT id, user_id, status, created_at, amount
FROM orders
WHERE user_id = 1001
  AND status = 1
ORDER BY created_at DESC
LIMIT 20;

一开始数据少,这条 SQL 看不出问题。等用户订单多了以后,数据库需要先筛出大量记录,再按 created_at 排序,接口就慢了。

MySQL ORDER BY 排序变慢的排查流程图,展示慢接口、查看计划、发现 filesort、设计联合索引和耗时下降

初步判断:先确认慢在排序还是筛选

我们先不要直接创建索引。第一步拆开看:筛选条件是谁,排序字段是谁,返回多少行。

  • user_id 是等值筛选。
  • status 是等值筛选。
  • created_at 是排序字段。
  • LIMIT 20 只要前 20 条。

如果索引能按 user_idstatus 定位,再顺着 created_at 的顺序取数据,数据库就不需要额外排序。反过来,如果索引只能筛选不能满足排序,就可能出现 Using filesort

动手验证:用 EXPLAIN 看 Extra

接着看执行计划:

EXPLAIN
SELECT id, user_id, status, created_at, amount
FROM orders
WHERE user_id = 1001
  AND status = 1
ORDER BY created_at DESC
LIMIT 20;

重点看几个字段:

  • key:实际使用了哪个索引。
  • rows:预计要扫描多少行。
  • Extra:是否出现 Using filesort

如果你看到类似结果:

key: idx_user_id
rows: 25800
Extra: Using where; Using filesort

这一步说明:MySQL 可能先用 user_id 找到一批数据,再按 created_at 做额外排序。数据越多,这个排序成本越明显。

定位原因:索引顺序和排序方向没对齐

现在可以定位到原因:现有索引只覆盖了部分筛选条件,没有把排序字段放进同一条可用路径里。

比如只有下面这个索引:

CREATE INDEX idx_user_id ON orders(user_id);

它能帮我们找到某个用户的订单,但不能保证这些订单已经按 statuscreated_at 的组合顺序排好。于是查询还要继续过滤和排序。

设计索引时要顺着 SQL 的读取路径看:

  • 先放等值筛选字段:user_idstatus
  • 再放排序字段:created_at
  • 如果分页需要稳定顺序,可以补上 id

修复方案:按筛选和排序设计联合索引

针对这条 SQL,可以创建这样的联合索引:

CREATE INDEX idx_user_status_created
ON orders(user_id, status, created_at DESC);

如果你的 MySQL 版本或团队规范不使用降序索引,也可以先使用:

CREATE INDEX idx_user_status_created
ON orders(user_id, status, created_at);

很多场景下 MySQL 仍然可以反向扫描索引拿到倒序结果。真正要不要显式写 DESC,要结合版本、SQL 和实际计划确认。

加索引后再看一次计划,理想变化是:

key: idx_user_status_created
rows: 20
Extra: Using where

这说明数据库可以沿着联合索引定位并按顺序取前 20 条,不再额外排序。

MySQL ORDER BY 联合索引优化流程图,展示筛选字段、排序字段、联合索引、减少扫描和耗时下降

验证结果:rows、Extra 和耗时一起看

最后不要只看 SQL 能不能跑,还要看三个信号:

  • rows 是否明显下降。
  • Extra 是否不再出现 Using filesort
  • 接口耗时是否在真实参数下稳定下降。

可以用同一组用户和状态做前后对比:

优化前:rows 25800,Extra 包含 Using filesort,耗时 1200ms
优化后:rows 20,Extra 不含 Using filesort,耗时 35ms

如果计划变好了但接口仍慢,就继续看是否有回表、网络传输、返回字段过多、连接池等待等其他问题。索引只是这一轮排查的核心,不代表所有慢都来自排序。

常见坑位和总结

1. 只给排序字段单独建索引

如果 SQL 有强筛选条件,单独给 created_at 建索引不一定合适。数据库可能按时间顺序扫很多不属于目标用户的数据,反而不稳。

2. 联合索引字段顺序随便放

等值筛选字段通常放前面,排序字段接在后面。字段顺序要贴近 SQL 的筛选和排序路径。

3. 分页排序没有稳定字段

如果多个订单的 created_at 完全相同,只按时间排序可能导致翻页时顺序抖动。可以考虑在排序里加上 id,并把它纳入索引设计。

4. 只在小数据量上验证

小表里很多问题看不出来。要用接近真实分布的数据验证,尤其是某些用户、店铺或租户的数据量明显更大时。

总结一下,ORDER BY 变慢的排查思路是:先确认筛选字段和排序字段,再用 EXPLAIN 看是否出现 Using filesort,接着按等值筛选加排序字段设计联合索引,最后用 rowsExtra 和接口耗时一起验证。这样排查,比盲目加索引更稳,也更容易解释优化为什么有效。

版本声明
本文转载于:17golang原创 如有侵犯,请联系study_golang@163.com删除
AI 提示词回归测试实战:小样本集、评分规则和上线前对比AI 提示词回归测试实战:小样本集、评分规则和上线前对比
上一篇
AI 提示词回归测试实战:小样本集、评分规则和上线前对比
前端详情页返回列表丢失滚动位置怎么办:从复现到恢复一步步排查
下一篇
前端详情页返回列表丢失滚动位置怎么办:从复现到恢复一步步排查
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    543次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    516次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    500次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    485次学习
查看更多
AI推荐
  • MiMo Code - 小米大模型团队开源的新一代 AI 编程助手
    MiMo Code
    MiMo Code 是小米大模型团队开源的新一代 AI 编程助手,面向开发者提供代码理解、生成与辅助开发能力,适合作为 AI 编程工具收藏和体验。
    42次使用
  • TRAE Work - 字节跳动推出的 AI 原生工作台
    TRAE Work
    TRAE AI IDE | 国内首款 AI 原生集成开发环境,深度集成 Doubao-1.5-pro 与 DeepSeek 模型,支持中文自然语言一键生成完整代码框架,实时预览前端效果并智能修复 BUG。首创 Builder 模式实现需求到代码的自动化开发,兼容 Windows/macOS 系统,官网下载即用。
    63次使用
  • MeloLab - 一站式 AI 音乐生成与编辑平台
    MeloLab
    MeloLab 是一款 AI 音乐生成工具,可根据文本创意生成歌曲、人声、混音、分轨和背景音乐,适合创作者快速制作音乐素材。
    52次使用
  • ChatExcel酷表:告别Excel难题,北大团队AI助手助您轻松处理数据
    ChatExcel酷表
    ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
    8706次使用
  • Any绘本:开源免费AI绘本创作工具深度解析
    Any绘本
    探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
    9115次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码