▲《50页性能分析案例总结》▲
长按二维码免费领取
Explain是一个非常有的命令,可以用来获取关于查询执行计划的信息,以及如何解释输出。Explain命令是查看查询优化器如何决定执行查询的主要方法。这个功能有一定的局限性,并不总是会说出真相,但是它的输出是可以获取的最好信息,值得花时间了解,可以学习到查询是如何执行的。
01
调用Explain
要使用Explain,只需在查询中的select关键字之前增加Explain这个词。MySQL会在查询上设置一个标记。当执行查询时,这个标记会使其返回关于在执行计划中每一步的信息,而不是执行它。
我们来简单看一下例子:可能是最简单的Explain结果
在查询中每个表在输出中只有一行。如果查询是两个表的联接,那么输出中将有两行。别名表单算为一个表。
02
Explain有两个主要的变种
Explain extended看起来和正常的explain行为一样,但它会告诉服务器“逆向编译”执行计划为一个select语句。可以通过紧接其后运行showwarnings看到这个生成的语句。这个语句直接来自执行计划,而不是原SQL语句,到这点上已经变成一个数据结构。大部分场景下,它都是优化过的,跟原语句不相同,可以学习查询优化器到底是如何转化语句的。
Explain partitions会显示查询将访问的分区,如果查询是基于分区表的话。一般认为增加explain时,MySQL语句不会执行查询,这是错误的。如果查询在from子句中包括子查询,那么MySQL实际上是会执行子查询,将其结果放在一个临时表中,然后完成外层查询优化。
前面简单解释了一下Explain可以做到的事情,但是Explain也有自身的一些限制:
03
重写非select查询
MySQL Explain只能解释select查询,并不会对存储过程调用和insert,update,delete或其他语句做解释。但是,我们可以重写这些非select语句来利用explain。为了利用explain,我们需要将这些语句转化成一个等价的访问所有相同列的select,所有需要的列必须在select列表,关联子句,或者where子句中。
04
Explain中的列
id列
这一列总是包含一个编号,标示select所属的行。数字越大越先执行,如果说数字一样大,那么就从上往下依次执行,id列为null的就表示这是一个结果集,不需要使用它来进行查询。
select_type列
这一列显示了对应行是简单还是复杂select。
常见的有:
table列
这一列显示了对应行正在访问查询的表名,如果查询使用了别名,那么这里显示的是别名,如果不涉及对数据表的操作,那么这显示为null,如果显示为尖括号括起来的就表示这个是临时表,后边的N就是执行计划中的id,表示结果来自于这个查询产生。如果是尖括号括起来的,与类似,也是一个临时表,表示这个结果来自于union查询的id为M,N的结果集。
type列
这一列显示了访问类型,即MySQL决定如何查找表中的行。
依次从好到差:system,const,eq_ref,ref,fulltext,ref_or_null,unique_subquery,index_subquery,range,index_merge,index,ALL,除了all之外,其他的type都可以使用到索引,除了index_merge之外,其他的type只可以用到一个索引
possible_keys列
查询可能使用到的索引都会在这里列出来。这个列表是优化过程早期创建的,因此有些罗列出来的索引有可能后续是没用的。
key列
显示了查询真正使用到的索引,select_type为index_merge时,这里可能出现两个以上的索引,其他的select_type这里只会出现一个。
如果该索引没有出现在possible_keys列中,那么MySQL选用它是出于另外的原因,比如选择了一个覆盖索引。
possible_keys揭示了哪一个索引能有助于高效地行查找,key显示了优化采用哪一个索引可以最小化查询成本。
key_len列
用于处理查询的索引长度,如果是单列索引,那就整个索引长度算进去,如果是多列索引,那么查询不一定都能使用到所有的列,具体使用到了多少个列的索引,这里就会计算进去,没有使用到的列,这里不会计算进去。
留意下这个列的值,算一下你的多列索引总长度就知道有没有使用到所有的列了。要注意,mysql的ICP特性使用到的索引不会计入其中。另外,key_len只计算where条件用到的索引长度,而排序和分组就算用到了索引,也不会计算到key_len中。
ref列
如果是使用的常数等值查询,这里会显示const,如果是连接查询,被驱动表的执行计划这里会显示驱动表的关联字段,如果是条件使用了表达式或者函数,或者条件列发生了内部隐式转换,这里可能显示为func
rows列
这里是执行计划中估算的扫描行数,不是精确值
extra列
这个列可以显示的信息非常多,有几十种,常用的有
除了这些之外,还有很多查询数据字典库,执行计划过程中就发现不可能存在结果的一些提示信息
filtered列
使用explain extended时会出现这个列,5.7之后的版本默认就有这个字段,不需要使用explain extended了。这个字段表示存储引擎返回的数据在server层过滤后,剩下多少满足查询的记录数量的比例,注意是百分比,不是具体记录数。
05
纵向表结构输出
在查询过程中,有时候信息太多的时候,横向输出会特别不容易读取,这时候,我们可以使用G将结果进行格式转换,将横向的表结构会转为使用纵向表结构输出,利于阅读。
这个格式化输出也可以用在select语句后
学习安排上!
码个资料送给你
▲《50页性能分析案例总结》▲
扫描下方二维码免费领取
上文内容不用于商业目的,如涉及知识产权问题,请权利人后台留言联系码同学小编,我们将立即处理。
“在看”点一点