EXPLAIN各个字段是什么意思

mysql使用explain xxx可以分析sql语句的性能,本文详细总结一下每个字段的意义

MYSQL 5.6.3以前只能EXPLAIN SELECT; MYSQL5.6.3以后就可以EXPLAIN SELECT,UPDATE,DELETE

概述

首先看一个简单的示例

mysql> explain select * from staff;
+----+-------------+-------+------+---------------+------+---------+------+------+-------+
| id | select_type | table | type | possible_keys | key  | key_len | ref  | rows | Extra |
+----+-------------+-------+------+---------------+------+---------+------+------+-------+
|  1 | SIMPLE      | staff | ALL  | NULL          | NULL | NULL    | NULL |    2 | NULL  |
+----+-------------+-------+------+---------------+------+---------+------+------+-------+
1 row in set

先总结下所有字段的意义

column含义
id查询序号
select_type查询类型
table表名
typejoin类型
prossible_keys可能会选择的索引
key实际选择的索引
key_len索引的长度
ref与索引作比较的列
rows要检索的行数(估算值)
Extra额外信息

再详细介绍每个字段的意义

1. id

SQL查询中的序列号

id列数字越大越先执行,如果说数字一样大,那么就从上往下依次执行

例如:

mysql> explain select * from (select * from ( select * from t3 where id=3952602) a) b;
+----+-------------+------------+--------+-------------------+---------+---------+------+------+-------+
| id | select_type | table      | type   | possible_keys     | key     | key_len | ref  | rows | Extra |
+----+-------------+------------+--------+-------------------+---------+---------+------+------+-------+
|  1 | PRIMARY     | <derived2> | system | NULL              | NULL    | NULL    | NULL |    1 |       |
|  2 | DERIVED     | <derived3> | system | NULL              | NULL    | NULL    | NULL |    1 |       |
|  3 | DERIVED     | t3         | const  | PRIMARY,idx_t3_id | PRIMARY | 4       |      |    1 |       |
+----+-------------+------------+--------+-------------------+---------+---------+------+------+-------+

sql语句从内到外按照id的顺序3 => 2 => 1依次执行

2. select_type

查询的类型,可以是下表的任何一种类型

select_type类型说明
SIMPLE简单SELECT(不使用UNION或子查询)
PRIMARY最外层的SELECT
UNIONUNION中第二个或之后的SELECT语句
DEPENDENT UNIONUNION中第二个或之后的SELECT语句取决于外面的查询
UNION RESULTUNION的结果
SUBQUERY子查询中的第一个SELECT
DEPENDENT SUBQUERY子查询中的第一个SELECT, 取决于外面的查询
DERIVED衍生表(FROM子句中的子查询)
MATERIALIZED物化子查询
UNCACHEABLE SUBQUERY结果集无法缓存的子查询,必须重新评估外部查询的每一行
UNCACHEABLE UNIONUNION中第二个或之后的SELECT,属于无法缓存的子查询

SIMPLE示例:

mysql> explain select * from t3 where id=3952602;
+----+-------------+-------+-------+-------------------+---------+---------+-------+------+-------+
| id | select_type | table | type  | possible_keys     | key     | key_len | ref   | rows | Extra |
+----+-------------+-------+-------+-------------------+---------+---------+-------+------+-------+
|  1 | SIMPLE      | t3    | const | PRIMARY,idx_t3_id | PRIMARY | 4       | const |    1 |       |
+----+-------------+-------+-------+-------------------+---------+---------+-------+------+-------+

PRIMARY示例:

mysql> explain select * from (select * from t3 where id=3952602) a ;
+----+-------------+------------+--------+-------------------+---------+---------+------+------+-------+
| id | select_type | table      | type   | possible_keys     | key     | key_len | ref  | rows | Extra |
+----+-------------+------------+--------+-------------------+---------+---------+------+------+-------+
|  1 | PRIMARY     | <derived2> | system | NULL              | NULL    | NULL    | NULL |    1 |       |
|  2 | DERIVED     | t3         | const  | PRIMARY,idx_t3_id | PRIMARY | 4       |      |    1 |       |
+----+-------------+------------+--------+-------------------+---------+---------+------+------+-------+

UNION、UNION RESULT示例:

mysql> explain select * from t3 where id=3952602 union all select * from t3 ;
+----+--------------+------------+-------+-------------------+---------+---------+-------+------+-------+
| id | select_type  | table      | type  | possible_keys     | key     | key_len | ref   | rows | Extra |
+----+--------------+------------+-------+-------------------+---------+---------+-------+------+-------+
|  1 | PRIMARY      | t3         | const | PRIMARY,idx_t3_id | PRIMARY | 4       | const |    1 |       |
|  2 | UNION        | t3         | ALL   | NULL              | NULL    | NULL    | NULL  | 1000 |       |
|NULL | UNION RESULT | <union1,2> | ALL   | NULL              | NULL    | NULL    | NULL  | NULL |       |
+----+--------------+------------+-------+-------------------+---------+---------+-------+------+-------+

3. table

查询的表名。不一定是实际存在的表。

可以为如下的值:

  • <unionM,N>: 引用idMN UNION后的结果。

  • <derivedN>: 引用idN的结果派生出的表。派生表可以是一个结果集,例如派生自FROM中子查询的结果。

  • <subqueryN>: 引用idN的子查询结果物化得到的表。即生成一个临时表保存子查询的结果。

例如:

mysql> explain select * from (select * from ( select * from t3 where id=3952602) a) b;
+----+-------------+------------+--------+-------------------+---------+---------+------+------+-------+
| id | select_type | table      | type   | possible_keys     | key     | key_len | ref  | rows | Extra |
+----+-------------+------------+--------+-------------------+---------+---------+------+------+-------+
|  1 | PRIMARY     | <derived2> | system | NULL              | NULL    | NULL    | NULL |    1 |       |
|  2 | DERIVED     | <derived3> | system | NULL              | NULL    | NULL    | NULL |    1 |       |
|  3 | DERIVED     | t3         | const  | PRIMARY,idx_t3_id | PRIMARY | 4       |      |    1 |       |
+----+-------------+------------+--------+-------------------+---------+---------+------+------+-------+

4. type(重要)

这是最重要的字段之一,显示查询使用了何种类型

下表从上到下性能依次递减

type说明
system表中只有一行数据或者是空表,这是const类型的一个特例。且只能用于myisammemory表。如果是Innodb引擎表,type列在这个情况通常都是all或者index
const最多只有一行记录匹配。当联合主键或唯一索引的所有字段跟常量值比较时,join类型为const。其他数据库也叫做唯一索引扫描
eq_ref多表join时,对于来自前面表的每一行,在当前表中只能找到一行。它用在一个索引的所有部分被联接使用并且索引是UNIQUEPRIMARY KEY
ref对于来自前面表的每一行,在此表的索引中可以匹配到多行。
fulltext使用全文索引的时候是这个类型。要注意,全文索引的优先级很高,若全文索引和普通索引同时存在时,mysql不管代价,优先选择使用全文索引
ref_or_null该联接类型如同ref,但是添加了MySQL可以专门搜索包含NULL值的行
index_merge表示查询使用了两个以上的索引,最后取交集或者并集,常见andor的条件使用了不同的索引
unique_subquery用于where中的in形式子查询,子查询返回不重复值唯一值,可以完全替换子查询,效率更高。 该类型替换了下面形式的IN子查询的refvalue IN (SELECT primary_key FROM single_table WHERE some_expr)
index_subquery该联接类型类似于unique_subquery。适用于非唯一索引,可以返回重复值。
range索引范围查询,常见于使用 =, <>, >, >=, <, <=, IS NULL, <=>, BETWEEN, IN()或者like等运算符的查询中。比如:SELECT * FROM tbl_name WHERE key_column BETWEEN 10 and 20;
index索引全表扫描,把索引从头到尾扫一遍。这里包含两种情况: 一种是查询使用了覆盖索引,那么它只需要扫描索引就可以获得数据,这个效率要比全表扫描要快,因为索引通常比数据表小,而且还能避免二次查询。在extra中显示Using index,反之,如果在索引上进行全表扫描,没有Using index的提示。
ALL全表扫描,性能最差

const示例:

mysql> explain select * from t3 where id=3952602;
+----+-------------+-------+-------+-------------------+---------+---------+-------+------+-------+
| id | select_type | table | type  | possible_keys     | key     | key_len | ref   | rows | Extra |
+----+-------------+-------+-------+-------------------+---------+---------+-------+------+-------+
|  1 | SIMPLE      | t3    | const | PRIMARY,idx_t3_id | PRIMARY | 4       | const |    1 |       |
+----+-------------+-------+-------+-------------------+---------+---------+-------+------+-------+

eq_ref示例:

mysql> create unique index  idx_t3_id on t3(id) ;
Query OK, 1000 rows affected (0.03 sec)
Records: 1000  Duplicates: 0  Warnings: 0
 
mysql> explain select * from t3,t4 where t3.id=t4.accountid;
+----+-------------+-------+--------+-------------------+-----------+---------+----------------------+------+-------+
| id | select_type | table | type   | possible_keys     | key       | key_len | ref                  | rows | Extra |
+----+-------------+-------+--------+-------------------+-----------+---------+----------------------+------+-------+
|  1 | SIMPLE      | t4    | ALL    | NULL              | NULL      | NULL    | NULL                 | 1000 |       |
|  1 | SIMPLE      | t3    | eq_ref | PRIMARY,idx_t3_id | idx_t3_id | 4       | dbatest.t4.accountid |    1 |       |
+----+-------------+-------+--------+-------------------+-----------+---------+----------------------+------+-------+

index_merge示例:

mysql> explain select * from t4 where id=3952602 or accountid=31754306 ;
+----+-------------+-------+-------------+----------------------------+----------------------------+---------+------+------+------------------------------------------------------+
| id | select_type | table | type        | possible_keys              | key                        | key_len | ref  | rows | Extra                                                |
+----+-------------+-------+-------------+----------------------------+----------------------------+---------+------+------+------------------------------------------------------+
|  1 | SIMPLE      | t4    | index_merge | idx_t4_id,idx_t4_accountid | idx_t4_id,idx_t4_accountid | 4,4     | NULL |    2 | Using union(idx_t4_id,idx_t4_accountid); Using where |
+----+-------------+-------+-------------+----------------------------+----------------------------+---------+------+------+------------------------------------------------------+

range示例:

mysql> explain SELECT * FROM t3 WHERE id BETWEEN 10 and 1000;
+----+-------------+-------+-------+-------------------+-----------+---------+------+------+-------------+
| id | select_type | table | type  | possible_keys     | key       | key_len | ref  | rows | Extra       |
+----+-------------+-------+-------+-------------------+-----------+---------+------+------+-------------+
|  1 | SIMPLE      | t3    | range | PRIMARY,idx_t3_id | idx_t3_id | 4       | NULL |    2 | Using where |
+----+-------------+-------+-------+-------------------+-----------+---------+------+------+-------------+

5. possible_keys

查询可能使用到的索引都会在这里列出来

6. key

查询真正使用到的索引。

select_typeindex_merge时,这里可能出现两个以上的索引,其他的select_type这里只会出现一个。

7. key_len

查询用到的索引长度(字节数)

如果是单列索引,那就整个索引长度算进去,如果是多列索引,那么查询不一定都能使用到所有的列,用多少算多少。

在不损失精确性的情况下,长度越短越好

key_len只计算where条件用到的索引长度,而排序和分组就算用到了索引,也不会计算到key_len

8. ref

和前方表连接的字段

如果是使用的常数等值查询,这里会显示const,如果是连接查询,被驱动表的执行计划这里会显示驱动表的关联字段,如果是条件使用了表达式或者函数,或者条件列发生了内部隐式转换,这里可能显示为func

9. rows(重要)

这是mysql估算的需要扫描的行数(不是精确值)。

这个值非常直观显示 SQL 的效率好坏, 原则上 rows 越少越好.

10. Extra(重要)

EXplain 中的很多额外的信息会在 Extra 字段显示, 常见的有以下几种内容:

Extra说明
Distinct一旦找到了与行相联合匹配的行,就不再搜索了。在select部分使用了distinc关键字会出现
Using filesort表示需额外的排序操作, 不能通过索引顺序达到排序效果。看到这个的时候,查询就需要优化了。
Using index“覆盖索引扫描”, 表示查询在索引树中就可查找所需数据, 不用扫描表数据文件, 往往说明性能不错
Using temporary查询有使用临时表, 一般出现于排序, 分组和多表 join 的情况, 查询效率不高, 建议优化.

以上就是explain各个字段的意义,这里一般做个进阶了解

在实际开发中,为了提升开发效率,我们可以选择搭配一些可视化SQL优化工具使用,可以参考我的这篇文章《小白也会用的SQL优化工具推荐》

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
引用\[1\]: EXPLAIN关键字用于模拟优化器执行SQL语句,分析查询语句的性能瓶颈。通过在SELECT语句之前加上EXPLAIN关键字,MySQL会返回该语句的执行计划。这个执行计划包含了查询的各个步骤,以及每个步骤的详细信息。\[2\]在使用EXPLAIN关键字时,如果FROM子句包含子查询,MySQL仍然会执行该子查询,并将结果放入临时表。这样可以更好地分析查询的性能。引用\[3\]在案例EXPLAIN语句返回了一个执行计划,其包含了查询的各个步骤的详细信息。在这个案例,查询语句是SELECT * FROM t_order ORDER BY express_type。执行计划字段列说明如下: - id: 表示查询的每个步骤的标识符,从1开始递增。 - select_type: 表示查询的类型。在这个案例,select_type为SIMPLE,表示这是一个简单的SELECT查询。 - table: 表示查询涉及的表名。 - type: 表示访问表的方式。在这个案例,type为ALL,表示全表扫描。 - possible_keys: 表示可能使用的索引。 - key: 表示实际使用的索引。 - key_len: 表示索引的长度。 - ref: 表示使用的索引的列或常数。 - rows: 表示扫描的行数。 - Extra: 表示额外的信息。在这个案例,Extra为Using filesort,表示需要进行文件排序。 通过分析这些字段列的信息,可以了解查询语句的执行计划,从而找出查询的性能瓶颈。 #### 引用[.reference_title] - *1* [一文看懂MySQLexplain关键字的作用](https://blog.csdn.net/qq_43332570/article/details/106860200)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [Explain关键字详解](https://blog.csdn.net/cczxcce/article/details/121440270)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [MySQL关键字EXPLAIN的用法及其案例](https://blog.csdn.net/u011863024/article/details/106818285)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值