听说你对 explain 很懂?
explain 所有人都应该很熟悉,通过它我们可以知道 SQL 是如何执行的,虽然不是 100%管用,但是至少大多数场景通过 explain 的输出结果我们能直观的看到执行计划的相关信息。
早一些的版本 explain 还只能查看select
语句,现在已经能支持delete
,update
,insert
,replace
了。
刚开始我想写这个的时候只是因为这个东西经常性不用就忘记,写了发现其实这个东西真的挺麻烦的,要把每个场景都整出来麻烦的很。
id
查询编号,如果没有子查询或者联合查询的话,就只有一条,如果是联合查询的话,那么会出现一条 id 为 null 的记录,并且标志查询结果,因为union
结果会放到临时表中,所以我们看到这里的表名是<union1,2>这种格式。
data:image/s3,"s3://crabby-images/f3fe5/f3fe5826b917ab8f43c170ae2c72631cfc1bd3e9" alt=""
select_type
关联类型,决定访问表的方式。
data:image/s3,"s3://crabby-images/c5024/c5024c166923742291e4d31f6caca7cdf2a640f1" alt=""
SIMPLE
简单查询,代表没有子查询或者union
。
PRIMARY
如果不是简单查询,那么最外层查询就会被标记成 PRIMARY。
UNION&UNION RESULT
从上图可以看出来了,包含联合查询,第一个被标记成了PRIMARY
,union 之后的查询被标记成UNION
,以及最后产生的UNION RESULT
DERIVED
用来标记出现在 from 里的子查询,这个结果会放入临时表中,也叫做派生表。
data:image/s3,"s3://crabby-images/f0fc7/f0fc7d20816615cfd47ce6f55477f31976dee364" alt=""
这个对于低版本的 Mysql 可能显示是这样的,高一点可能你看到的还是 PRIMARY,因为被 Mysql 优化了。我换一个版本的 Mysql 和 SQL 执行可以验证到这个结果。
data:image/s3,"s3://crabby-images/6fe24/6fe2490b07c6b0c9e97f69e7cd65282b22b44ef6" alt=""
SUBQUERY
不在 from 里的子查询。
data:image/s3,"s3://crabby-images/d3274/d3274bb28545f4d5cb01ae80aeb671430beb9148" alt=""
DEPENDENT
代表关联子查询(子查询使用了外部查询包含的列),和UNION
,SUBQUERY
组合产生不同的结果。
data:image/s3,"s3://crabby-images/f2bd6/f2bd6a5e9aaec51a0d933e0c731d851dc4cfff12" alt=""
UNCACHEABLE
代表不能缓存的子查询,也可以和UNION
,SUBQUERY
组合产生不同的结果。
data:image/s3,"s3://crabby-images/7b83c/7b83c5696b73e7a89ab059ff5dfda9e36f965d64" alt=""
MATERIALIZED
物化子查询是 Mysql 对子查询的优化,第一次执行子查询时会将结果保存到临时表,物化子查询只需要执行一次。
比如上述 DERIVED 就是物化的一种体现,与之对应的就是 DEPENDENT,每次子查询都需要重新调用。
这个结果无法直观的看出来,可以用FORMAT=JSON
命令查看materialized_from_subquery
字段。
data:image/s3,"s3://crabby-images/e6a9d/e6a9d3869e758e348a50d01e9d42b02c9a757b32" alt=""
table
显示表名,从上述的一些图中可以观察到 UNION_RESULT 和 DERIVED 显示的表名都有一些自己的命名规则。
比如 UNION_RESULT 产生的是<unionM,N>,DERIVED 产生的是<derivedN>。
partitions
数据的分区信息,没有分区忽略就好了。
type
关联类型,决定通过什么方式找到每一行数据。以下按照速度由快到慢。
system>const>eq_ref>ref>fulltext>ref_or_null>index_merge>unique_subquery>index_subquery>range>index>ALL。
data:image/s3,"s3://crabby-images/9c4b7/9c4b768c5cef06c95ca49f61638d0185e377800b" alt=""
system&const
这通常是最快的查找方式,代表 Mysql 通过优化最终转换成常量查询,最常规的做法就是直接通过主键或者唯一索引查询。
data:image/s3,"s3://crabby-images/bcae9/bcae9013ca05a22400fe7ff7a5e0c70723079439" alt=""
而 system 是 const 的一个特例(只有一行数据的系统表),随便找一张系统表,就插入一条数据就可以看到 system 了。
data:image/s3,"s3://crabby-images/22ea7/22ea7101fdefa5e60297ec0d2e79465f925546f7" alt=""
eq_ref
通常通过主键索引或者唯一索引查询时会看到 eq_ref,它最多只返回一条数据。user_id
是唯一索引,为了测试就关联以下主键索引。
data:image/s3,"s3://crabby-images/e22cc/e22cceae016cc09d4f44e0a8cf8063efd87089bd" alt=""
ref
也是通过索引查找,但是和 eq_ref 不同,ref 可能匹配到多条符合条件的数据,比如最左前缀匹配或者不是主键和唯一索引。
最简单的办法,随便查一个普通索引就可以看到。
data:image/s3,"s3://crabby-images/a9483/a948354ec50cea03921dc864f281623f91b153c2" alt=""
fulltext
使用 FULLTEXT 索引
ref_or_null
和 ref 类似,但是还要进行一次查询找到 NULL 的数据。
这相当于是对于 IS NULL 查询的优化,如果表数据量太少的话,你或许能看到这里类型是全表扫描。
data:image/s3,"s3://crabby-images/0391a/0391acb27763323c296911d17d2bee920771b21f" alt=""
index_merge
索引合并是在 Mysql5.1 之后引入的,就像下面的一个 OR 查询,按照原来的想法要么用 name 的索引,要么就是用 age 的索引,有了索引合并就不一样了。
对于这种单表查询(无法跨表合并)用到了多个索引的情况,每个索引都可能返回一个结果,Mysql 会对结果进行取并集、交集,这就是索引合并了。
data:image/s3,"s3://crabby-images/a74b1/a74b187153ffa0265a49315ab3a6f341cb3eea0e" alt=""
unique_subquery
按照官方文档所说,unique_subquery 只是 eq_ref 的一个特例,对于下图中这种in
的语句查询会出现以提高查询效率。
由于 Mysql 会对 select 进行优化,基本无法出现这个场景,只能用 update 这种语句了。
data:image/s3,"s3://crabby-images/45851/458518466d3443ccbeea17d877909da15f8afe2a" alt=""
index_subquery
和 unique_subquery 类似,只是针对的是非唯一索引。
data:image/s3,"s3://crabby-images/09821/09821279626a745cf68a4dd09809d076ce7ea7b3" alt=""
range
看名字就知道,范围查询,其实就是带有限制条件的索引扫描。
常见的范围查询比如between and
,>,<,like,in 都有可能出现 range。
data:image/s3,"s3://crabby-images/dba5a/dba5a692b60e41cfcdda9f4c0b925bf580df9f3d" alt=""
index
跟全表扫描类似,只是扫表是按照索引顺序进行。
ALL
全表扫描,没啥好说的。
possible_keys
可以使用哪些索引。
key
实际决定使用哪个索引。
key_len
索引字段的可能最大长度,不是表中实际数据使用的长度。
ref
表示 key 展示的索引实际使用的列或者常量。
rows
查询数据需要读取的行数,只是一个预估的数值,但是能很直观的看出 SQL 的优劣了。
filtered
5.1 版本之后新增字段,表示针对符合查询条件的记录数的百分比估算,用 rows 和 filtered 相乘可以计算出关联表的行数。
Extra
解析查询的附加额外信息,这个太多了,有兴趣可以自己看官方文档,只列举一些常见的。
Using index
使用覆盖索引。
Using index condition
使用索引下推,索引下推简单来说就是加上了条件筛选,减少了回表的操作。
data:image/s3,"s3://crabby-images/7275a/7275af5b98563dd17b3dc3b63fe114ed1d6681b6" alt=""
Using temporary
排序使用了临时表。
Using filesort
使用外部索引文件排序,但是不能从这里看出是内存还是磁盘排序,我们只能知道更消耗性能。
Using where
where 过滤,没啥好说的。
Zero limit
除非你写个 LIMIT 0。
Using sort_union(), Using union(), sing intersect()
使用了索引合并,参看上文。
总结
data:image/s3,"s3://crabby-images/a1eda/a1eda9d82a8e687d785722cdd187b36a3e990bbb" alt=""
版权声明: 本文为 InfoQ 作者【艾小仙】的原创文章。
原文链接:【http://xie.infoq.cn/article/2b77afa12e2afa710fcf1aee3】。文章转载请联系作者。
评论