你是否也遇到过这样的情况:一条 SQL 平时运行得飞快,某一天忽然变得异常缓慢。翻出慢查询日志,找到了那条语句,却完全搞不清楚它为何变慢。尝试加个索引?没效果。换个写法?依然无效。最后只能重启数据库碰碰运气。
这种“凭感觉优化”之所以效果不佳,是因为缺少数据库的一份“自检报告”——它正是执行计划。
执行计划是数据库在执行 SQL 之前,提前展示给你的一份“作战方案”:它明确告诉你打算通过什么方式检索数据、使用哪些索引、预估扫描多少行记录,以及还需要执行哪些额外操作。学会解读执行计划,就能从“猜测”转变为“观察”,优化不再是玄学。
下面拆解执行计划中最核心的五个字段,理解它们的含义,便能快速定位慢查询的关键问题。
type:访问方式,性能的“红绿灯”
type表示数据库如何访问表中的数据。从最优到最差依次为:system > const > eq_ref > ref > range > index > ALL。
可以把它类比成开车上路的效率等级:
const:走专属快速通道,一杆直达(通过主键或唯一索引精确命中唯一一行)。ref:走普通城市主干道,稍慢但可接受(通过普通索引命中多行)。range:在主干道上遇到红绿灯,需要走走停停(索引范围扫描,如BETWEEN、>、<)。index:在辅路上缓慢挪动(全索引扫描,比全表扫描好一些,但仍有优化空间)。ALL:堵在路上,几乎停滞(全表扫描,必须优化)。
一个值得记住的判断标准:看到ALL或index,基本可以认定索引设计存在缺陷或没有可用索引。
key_len:复合索引使用了多少列
对于复合索引(a,b,c),key_len告诉你实际使用了多少列。例如,一个INT字段占4字节,DATE占3字节,VARCHAR根据字符集计算(通常utf8mb4每字符4字节,再加2字节长度标识)。如果索引定义总长为50字节,但key_len只有4,说明只用了第一列。
这个判断不需要死记公式,而是通过对比索引定义和key_len数值,就能知道查询条件是否命中了索引前缀、有没有跳过中间列。如果key_len偏小,往往是因为查询条件没有写全索引列,或者违背了最左匹配原则。
rows:估算需要扫描的行数
rows是优化器根据统计信息估算的需要扫描的行数。它是一个相对值,并非精确值,但其量级决定了查询成本。
判断标准:rows越大,通常性能越差。如果rows接近全表总行数,却还在使用索引,说明索引的选择性极低(例如只建立在性别这类字段上),优化器可能选错了方向。
filtered:索引筛选后剩余的比例
filtered表示存储引擎返回的行中,满足剩余WHERE条件的比例。100%是最理想的情况,意味着索引已经精准定位,无需额外过滤;10%则表明索引只筛掉了90%,回表后还要再过滤掉大部分数据,通常是因为索引列选择性差,或者查询条件中包含不在索引中的过滤字段。
诊断时注意:当filtered较低时,应考虑扩展索引,把过滤字段也包含进来,或者调整索引顺序。
Extra:额外的“小动作”
Extra列中藏着数据库在执行过程中需要做的额外操作,有些是好事,有些是坏事。
Using index:覆盖索引,无需回表 ✅Using index condition:索引条件下推,提前过滤,减少了回表 ✅Using where:需要回表后再过滤 ⚠️Using temporary:使用了临时表,常见于GROUP BY未走索引 ❌Using filesort:文件排序,常见于ORDER BY未走索引 ❌Using join buffer:JOIN 未使用索引 ❌
这些提示直接指明了优化方向:看到temporary就去给GROUP BY列加索引;看到filesort就去给ORDER BY列建索引;看到join buffer就去检查连接条件是否有索引。
为了更直观地理解这些字段如何配合,下面来看一个简化的诊断流程。
假设有一条慢查询,执行EXPLAIN后得到输出结果。不需要逐字逐句分析,而是按顺序问自己三个问题:
第一问:type是什么?
如果是ALL或index,问题根源在于访问方式过于原始。大概率是没有索引或索引未生效。先去检查WHERE条件涉及的列是否有索引,以及是否存在隐式类型转换、函数包裹索引列等导致失效的原因。
第二问:key_len是否合理?
对照创建的复合索引定义,看key_len是否覆盖了期望的列数。如果明显偏小,说明查询条件未用到索引前缀,需要调整索引列顺序或补全查询条件。
第三问:Extra里是否存在temporary或filesort?
如果有,说明GROUP BY或ORDER BY没有走索引。去检查这些列是否在索引中,以及索引顺序是否匹配排序要求。
这三个问题走完,80%的慢查询都能找到病因。剩下的20%通常与数据分布、统计信息过时有关,此时再配合ANALYZE TABLE更新统计信息,或者在生产环境用EXPLAIN ANALYZE查看真实执行数据。
从执行计划到优化动作,核心逻辑并不是堆砌索引,而是先读懂数据库给出的反馈,再有针对性地进行调整。type告诉你“怎么查”,key_len告诉你“用了多少列”,rows和filtered告诉你“代价有多大”,Extra告诉你“额外负担”。把这五个字段串联起来,就能在几十秒内判断一条 SQL 的健康状况,并快速锁定问题所在。
下次遇到慢查询时,别再盲目加索引了。先跑一遍EXPLAIN,让数据库告诉你它真正需要什么。

