🎯 EXPLAIN 关键指标与含义
| 指标 (Field) | 含义 (Meaning) | 优化关注点 | 理想值 |
|---|---|---|---|
| id | 查询中每个 SELECT 语句的编号。 | 用于识别子查询、联合查询或衍生表(Derived Table)的执行顺序。编号越大越先执行,相同编号按顺序执行。 | 越高越好(指查询的层级/复杂性)。 |
| select_type | 查询的类型,例如是简单查询、子查询、联合查询等。 | 识别查询的复杂性。不同的类型可能需要不同的优化策略。 | SIMPLE (最简单) |
| table | 正在访问的表名。 | 标识查询针对哪个表。 | 实际的表名。 |
| type | 连接类型 (Connection Type) 或 访问类型 (Access Type) ,这是最重要的指标之一。它表明 MySQL 如何查找表中的行。 | 至关重要! 决定了查询效率。从好到差的顺序大致是:system/const > eq_ref > ref > range > index > ALL。 |
const , eq_ref , ref , range |
| possible_keys | 可能被用于查找的索引列表。 | 提示您可以创建或检查哪些索引。 | 至少有一个或多个索引。 |
| key | 实际被选用的索引 。如果为 NULL,则表示没有使用索引。 |
检查 MySQL 是否选择了最佳索引。如果 possible_keys 有值而 key 为 NULL,需要优化。 |
一个合适的索引名。 |
| key_len | 实际使用的索引长度(字节)。 | 用于判断联合索引中使用了多少字段。key_len 越小,说明索引覆盖越精确,效率越高。 |
尽可能短,能满足查询需求即可。 |
| ref | 指示与索引的哪一列(或常量)进行比较。 | 常量 (const)、来自其他表的列 (db.table.column)。 |
const 或指向主键/唯一键的列。 |
| rows | MySQL 估计需要检查的行数。 | 至关重要! rows 越少越好,它是衡量查询性能的核心指标之一。 |
越小越好。 |
| filtered | 表示通过当前表条件过滤后,剩余的行数占总扫描行数 (rows) 的百分比。 |
filtered 越高越好 (接近 100%),说明查询条件过滤掉了大部分行。 |
100.00 |
| Extra | 额外信息,包含不适合在其他列中显示的信息,通常是非常关键的优化提示。 | 检查是否有负面 信息,如 Using temporary、Using filesort。追求正面 信息,如 Using index、Using where。 |
Using index, Using where |
⚠️ 特别关注的指标和值
在进行查询优化时,应将注意力集中在以下几个方面:
1. type 访问类型 (Access Type)
这是判断查询性能最重要且最直接的指标。
-
🏆 优秀类型:
const/system: 命中主键或唯一索引,且是常量值比较。极快。eq_ref: 联接使用唯一索引。常用于联接操作,非常好。ref: 联接使用非唯一索引。良好。
-
👍 可接受类型:
range: 索引范围扫描,如BETWEEN、<、>、IN等操作。一般。index: 全索引扫描,比ALL好,因为它只扫描索引树。较差。
-
👎 差劲类型 (需要优化):
ALL: 全表扫描。这是性能最低的类型,应尽力避免,通常意味着缺失索引或索引使用不当。
2. rows 扫描行数
- 原则: 扫描的行数越少越好。如果
rows值很大,即使type不是ALL,也可能表明查询效率不高。
3. Extra 额外信息
您应该特别注意并消除以下负面信息:
Using filesort: 表示 MySQL 对结果集进行了外部排序(在内存或磁盘上),通常是低效的 。应尝试在ORDER BY的列上创建索引来消除。Using temporary: 表示 MySQL 需要创建一个临时表来处理查询,通常是低效的 。常见于GROUP BY或ORDER BY与不同表联接字段混合使用时。
您应该追求以下正面信息:
Using index(覆盖索引): 表示查询所需的所有数据都可以在索引中找到,而无需回表 (Full Index Scan),性能极佳。Using where: 表示使用了WHERE条件对数据进行了过滤。Using index condition(ICP): 表示在存储引擎层而非 Server 层进行索引条件判断,性能优化。
总结来说,一个优化的查询应该努力达到:type 接近 const/eq_ref/ref/range ,rows 值尽可能小 ,并且 Extra 中不包含 Using filesort 和 Using temporary ,最好能实现 Using index。