你真的会用Mysql的explain吗


Posted in MySQL onMarch 31, 2022

引言

数据库性能优化是每个后端程序猿必备的基础技能之一,而Mysql中的explain堪称Mysql的性能优化分析神器,我们可以通过它来分析SQL语句的对应的执行计划在Mysql底层到底是如何执行的,它对于我们评估SQL的执行效率以及确定Mysql的性能优化方向具有重要的意义。但是很多同学对于如何根据explain对已有SQL进行深度的执行分析还是丈二和尚摸不着头脑,因此本文详细阐述通过explain分析定位数据库性能问题。

explain基础

对于每个SQL来说,当它被客户端发送到Mysql服务端之后,会经过Mysql的优化器部件的分析,主要包括一些特殊的处理、执行顺序的改变以确保最优的执行效率,最终生成对应的执行计划。所谓的执行计划,实际就是在存储引擎层面如何获取数据的,是通过索引获取数据还是进行全表扫描获取数据,获取到数据后需不需要回表,等等,简单理解就是Mysql获取数据的过程。

接下来我们来详细看下,这个explain到底是何方神圣,为什么能指导我们进行性能优化。当我们执行如下语句:

explain SELECT * FROM user_info where NAME='mufeng'explain SELECT * FROM user_info where NAME='mufeng'

执行explain语句之后,我们会得到如下的执行结果,这个类似数据库表的12个字段实际上就是对Mysql执行怎样的执行计划的详细描述。下面我们来好好研究下这12个字段分别代表什么意思,只有搞清楚它们的含义,我们才能明确Mysql到底是怎么执行数据查询的。

你真的会用Mysql的explain吗

你真的会用Mysql的explain吗

1、id

实际上每次select查询都会对应一个id,它代表着SQL执行的顺序,如果id值越大,说明对应的SQL语句执行的优先级越高。在一些复杂的查询SQL语句中常常包含一些子查询,那么id序号就会递增,如果出现嵌套查询,我们可以发现最里层的查询对应的id最大,因此也优先被执行。

你真的会用Mysql的explain吗

如上图所示,SQL查询语句中,第一个执行计划的id为1,第二个执行计划的id为2,id为1的执行计划对应的table为order,id为2的执行计划对应的table是user_info,结合SQL语句,我们知道先执行子查询select id from user_info,而后再执行关于表order的数据查询。

2、select_type

select_type表示的执行计划的对应的查询是什么类型,常见的查询类型主要包括普通查询、联合查询以及子查询等。SIMPLE(查询语句为简单的查询不包含子查询)、PRIMARY(当查询语句中包含子查询的时候,对应最外层的查询类型)、UNION(union之后出现的select语句对应的查询类型会标记此类型)、SUBQUERY(子查询会被标记为此类型)、DEPENDENT SUBQUERY(取决于外面的查询 )。

你真的会用Mysql的explain吗

3、table

table代表表名称,表示要查询哪张表。当然不一定是真实的表的名称,也可能是表的别名或者临时表。

4、partitions

partitions代表的是分区的概念,表示在进行查询时,如果对应的表存在分区表,那么这里就会显示具体的分区信息。

5、type

type是非常核心的属性,需要重点掌握。它表示的是当前通过什么样的方式对数据库表进行分访问。

你真的会用Mysql的explain吗

(1)system

该表只有一行(相当于系统表),数据量很小,查询速度很快,system是const类型的特例。

(2)const

如果type是const,说明在进行数据查询的时候,命中了primary key或唯一索引,此类数据查询速度非常快。

你真的会用Mysql的explain吗

(3)eq_ref

在进行数据查询的过程中,如果SQL语句中在表连接情况下可以基于聚簇索引或者非null值的唯一索引记性数据扫描,那么此时type对应的值就会显示为eq_ref。

(4)ref

数据查询的时候如果命中的索引是二级索引不是唯一索引,测试查询速度也会很快,但是type是ref。另外如果是多字段的联合索引,那么根据最左匹配原则,从联合索引的最左侧开始连续多个列的字段进行等值比较也是ref的类型。

你真的会用Mysql的explain吗

(5)ref_or_null

这种连接类型类似于 ref,区别在于 MySQL会额外搜索包含NULL值的行。

(7)unique_subquery

在where条件中的关于in的子查询条件集合

(8)index_subquery

区别于unique_subquery,用于非唯一索引,可以返回重复值。

(9)range

使用索引进行行数据检索,只对指定范围内的行数据进行检索。换句话说就是针对一个有索引的字段,在指定范围中检索数据。在where语句中使用 bettween...and、<、>、<=、in 等条件查询 type 都是 range。

你真的会用Mysql的explain吗

(10)index

Index 与ALL 其实都是读全表,区别在于index是遍历索引树读取,而ALL是从硬盘中读取。

(11)all

遍历全表进行数据匹配,此时的数据查询性能最差。

6、possible_keys

表示哪些索引可以被Mysql的优化器进行选择,也就是索引候选者有哪些。

7、key

在possible_keys中实际选择的索引

8、key_len

表示索引的长度,和实际的字段属性以及是否为null都有关系。

9、ref

你真的会用Mysql的explain吗

当使用字段进行常量等值查询时ref此处为const,当查询条件中使用了表达式或者函数则ref显示为func,则其他的显示为null。

10、rows

rows列显示MySQL认为它执行查询时必须检查的行数。行数越少,效率越高!

11、filtered

filtered 这个是一个百分比的值,表里符合条件的记录数的百分比。简单点说,这个字段表示存储引擎返回的数据在经过过滤后,剩下的记录满足条件的记录数量的比例。

12、extra

在其他列不显示额外信息在此列进行展示。

(1)Using index

在进行数据查询的时候,数据库使用了覆盖索引,就是查询的列被索引覆盖,使用到覆盖索引查询速度会非常快。不是使用select * ,而是使用select phone_number,就会用到覆盖索引。

你真的会用Mysql的explain吗

(2)Using where

 查询时未找到可用的索引,进而通过where条件过滤获取所需数据,但要注意的是并不是所有带where语句的查询都会显示Using where。

你真的会用Mysql的explain吗

(3)Using temporary

表示查询后结果需要使用临时表来存储,一般在排序或者分组查询时用到。

(4)Using filesort

此类型表示无法利用索引完成指定的排序操作,也就是ORDER BY的字段实际没有索引,因此此类SQL是需要进行优化的。

explain分析实战

上文中我们阐述了explain在分析SQL语句时,可以通过12个属性来分析SQL的大致执行过程,并以此来判断SQL存在的性能问题。那么接下来我们通过一个实际的例子,来具体看下如何结合explain来实现SQL的性能分析。

其实所谓的Mysql性能问题,大部分都指的是平台出现了慢查询问题。慢查询实际上是可以通过配置进行记录的,把执行时间超过某个设定的阈值的sql都记录下来,当出现问题的时候可以通过记录的慢查询日志进行问题的定位。但是有的时候,出现大量慢查询会导致数据库连接被占满,导致整个平台的出现异常。

实际上我们在产品评价表product_evaluation中是建立了索引的,正常来说应该是可以使用到对应的索引字段进行查询的。但是实际上查询耗时有几十秒的时间,远远超过我们的预期。那我们猜测是不是由于某种原因导致Mysql优化器没有选择对应的索引进行数据检索,最后造成慢查询的发生。到底执行计划是怎样的,还是得借助于explain来看下。

你真的会用Mysql的explain吗

如上文所说,虽然explain有12个字段属性帮助我们进行执行计划的分析,但是实际上常用的核心字段也就几个。我们可以看的出来在possible_key中实际上包含了我们设置的索引的,但是实际上Mysql却选择了PRIMARY作为其实际使用的。那么问题来了,为什么明明设置了索引,但是实际并没有用上,被Mysql吃了吗?另外为什么之前的业务中没有出现这个问题,而现在出现了?我们需要进行进一步的分析。

我们所建立的idx_evaluation_type实际上是一个二级索引(叶子节点是主键id),对于数千万一张的大表来说,实际上这个二级索引也是非常大的,而且这个字段本身的值就三个,变化不大。因此Mysql的优化器在分析这个SQL的时候发现,如果按照SQL中的索引来获取数据后再根据where条件进行筛选,筛选后的数据还需要回表到聚簇索引中获取实际的数据。

假如通过二级索引筛选出来的数据有几万条,而后还需要进行排序,这些操作都是基于临时磁盘我恩建进行的,Mysql判断这种方式的性能可能会很差,因此优化器放弃了原有的数据查询方式,直接通过主键id对应的聚簇索引来进行数据的获取,因为id本身就是有序的。

你真的会用Mysql的explain吗

那么知道了查询慢的原因,我们应该怎么进行优化呢?实际上可以在SQL语句中增加force idnex,强制Mysql使用我们设置的二级索引。

SELECT * FROM product_evaluation force index(idx_product_id)WHERE product_id =1 and evaluation_type='GOOD'  ORDER BY id desc LIMIT 200SELECT * FROM product_evaluation force index(idx_product_id)WHERE product_id =1 and evaluation_type='GOOD'  ORDER BY id desc LIMIT 200

总结

通过上文对于explain使用的介绍,大家在遇到慢SQL问题的时候,可以先通过explain来进行初步的分析,主要明确SQL在Mysql中实际的执行过程是怎样的,如果查询字段没有索引则增加索引,如果有索引就要分析为什么没有用到索引。只要明确具体的执行过程,我们才能确定具体的查询优化方案。

到此这篇关于Mysql中explain的文章就介绍到这了,更多相关Mysql explain用法内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!


Tags in this post...

MySQL 相关文章推荐
MySQL root密码的重置方法
Apr 21 MySQL
浅谈Mysql多表连接查询的执行细节
Apr 24 MySQL
mysql 8.0.24 安装配置方法图文教程
May 12 MySQL
MySQL 数据恢复的多种方法汇总
Jun 21 MySQL
MySQL命令无法输入中文问题的解决方式
Aug 30 MySQL
MyBatis 动态SQL全面详解
Oct 05 MySQL
Mysql数据库手动及定时备份步骤
Nov 07 MySQL
mysql的数据压缩性能对比详情
Nov 07 MySQL
mysql的单列多值存储实例详解
Apr 05 MySQL
MySQL中一条SQL查询语句是如何执行的
Apr 08 MySQL
MySQL数据库如何使用Shell进行连接
Apr 12 MySQL
MySQL 逻辑备份 into outfile
May 15 MySQL
MySQL限制查询和数据排序介绍
MySQL学习必备条件查询数据
mysql中数据库覆盖导入的几种方式总结
Mysql如何实现不存在则插入,存在则更新
Mar 25 #MySQL
MySQL插入数据与查询数据
mysql insert 存在即不插入语法说明
Mar 25 #MySQL
MySQL创建表操作命令分享
You might like
Laravel5中contracts详解
2015/03/02 PHP
在Mac OS的PHP环境下安装配置MemCache的全过程解析
2016/02/15 PHP
laravel创建类似ThinPHP中functions.php的全局函数
2016/11/26 PHP
php PDO判断连接是否可用的实现方法
2017/04/03 PHP
腾讯的ip接口 方便获取当前用户的ip地理位置
2010/11/25 Javascript
jQuery.Validate验证库的使用介绍
2013/04/26 Javascript
JavaScript获取onclick、onchange等事件值的代码
2013/07/22 Javascript
jquery定时滑出可最小化的底部提示层特效代码
2013/10/02 Javascript
Jquery实现自定义tooltip示例代码
2014/02/12 Javascript
jQuery 浮动导航菜单适合购物商品类型的网站
2014/09/09 Javascript
sails框架的学习指南
2014/12/22 Javascript
实例浅析js的this
2016/12/11 Javascript
BootStrap表单验证实例代码
2017/01/13 Javascript
Angular4实现动态添加删除表单输入框功能
2017/08/11 Javascript
解决Linux无法正常安装与卸载Node.js的方法
2018/01/19 Javascript
Vue-cli项目获取本地json文件数据的实例
2018/03/07 Javascript
使用vue2实现购物车和地址选配功能
2018/03/29 Javascript
Vue.set()动态的新增与修改数据,触发视图更新的方法
2018/09/15 Javascript
用Python的urllib库提交WEB表单
2009/02/24 Python
python如何拆分含有多种分隔符的字符串
2018/03/20 Python
解决pycharm remote deployment 配置的问题
2019/06/27 Python
Python使用Pandas对csv文件进行数据处理的方法
2019/08/01 Python
python发qq消息轰炸虐狗好友思路详解(完整代码)
2020/02/15 Python
Tensorflow实现将标签变为one-hot形式
2020/05/22 Python
一波HTML5 Canvas基础绘图实例代码集合
2016/02/28 HTML / CSS
成人教育自我鉴定
2013/11/01 职场文书
煤矿班组长竞聘书
2014/03/31 职场文书
高中升旗仪式演讲稿
2014/09/09 职场文书
党员贯彻十八大精神思想汇报范文
2014/10/25 职场文书
四年级数学上册教学计划
2015/01/20 职场文书
邀请书模板
2015/02/02 职场文书
2016五一手机促销广告语
2016/01/28 职场文书
高三数学复习备考教学反思
2016/02/18 职场文书
使用Html+Css实现简易导航栏功能(导航栏遇到鼠标切换背景颜色)
2021/04/07 HTML / CSS
python 自动化偷懒的四个实用操作
2021/04/11 Python
修改并编译golang源码的操作步骤
2021/07/25 Golang