当前位置:网站首页>MySQL - Explain explanation
MySQL - Explain explanation
2022-08-04 12:13:00 【no object pointer】
1. 前言
Server version: 5.7.36 MySQL Community Server (GPL)
mysql官方文档:https://dev.mysql.com/doc/refman/5.7/en/explain-output.html
explain(执行计划),使用 explain 关键字可以模拟优化器执行 sql 查询语句,从而知道 MySQL 是如何处理 sql 语句.explain 主要用于分析查询语句或表结构的性能瓶颈.
explain select * from user where phone = '15233658888';
explain 输出内容大致如下:
2. explain的作用
通过 explain + sql 语句可以知道如下内容:
表的读取顺序(对应id)
数据读取操作的操作类型(对应select_type)
哪些索引可以使用(对应possible_keys)
哪些索引被实际使用(对应key)
表直接的引用(对应ref)
每张表有多少行被优化器查询(对应rows)
3. explain的内容
3.1 id
select查询的序列号,包含一组数字,表示查询中执行select子句或操作表的顺序,This field is usually combined with tableField collocation to analyze.
id 相同时,执行顺序由上至下
如果是子查询,id 的序号会递增,id 值越大优先级越高,越先被执行
id 如果相同,可以认为是一组,从上往下顺序执行;在所有组中,id 值越大,优先级越高,越先执行
3.2 select_type
查询的类型,主要用于区别普通查询、联合查询、子查询等复杂的查询.其值主要有六个:
- SIMPLE
简单的 select 查询,查询中不包含子查询或 union 查询.
- PRIMARY
查询中若包含任何复杂的子部分,最外层查询为 PRIMARY,也就是最后加载的就是 PRIMARY.
- UNION
此查询是 UNION 中的第二个或后面的SELECT语句
若第二个 select 出现在 union 后,则被标记为 UNION,若 union 包含在 from 子句的子查询中,外层 select 将被标记为 DERIVED.
- DEPENDENT UNION
UNION 中的第二个或后面的SELECT语句,取决于外面的查询.
- UNION RESULT
UNION的结果.
- SUBQUERY
子查询中的第一个SELECT,结果不依赖于外部查询
在 select 或 where 列表中包含了子查询,就为被标记为 SUBQUERY.
- DERIVED
在 from 列表中包含的子查询会被标记为 DERIVED,MySQL 会递归执行这些子查询,将结果放在临时表中.
3.3 table
显示这一步所访问数据库中表名称(显示这一行的数据是关于哪张表的),有时不是真实的表名字,可能是简称,例如tab_a,tab_a,也可能是第几步执行的结果的简称
3.4 partitions
官方定义为The matching partitions(匹配的分区),对于非分区表值为null.
3.5 type
Indicates the type of access used by the query,type There are eight main values,该值表示查询的 sql 语句好坏,从最好到最差依次为:
NULL > system > const > eq_ref > ref > range > index > ALL
- NULL
MySQL 在优化过程中分解语句,执行时甚至不用访问表或索引,例如从一个索引列里选取最小值可以通过单独索引查找完成.
- system
system 是 const 的特例,表里只有一条元组匹配时为 system.
- const
MySQL 能对查询的某部分进行优化并将其转化成一个常量(可以看show warnings 的结果).用于 primary key 或 unique key 的所有列与常数比较时,表最多有一个匹配行,读取1次,速度比较快.
- eq_ref
唯一性索引扫描,对于每个索引键,表中只有一条记录与之匹配.常见于主键或唯一索引扫描;
primary key 或 unique key 索引的所有部分被连接使用 ,最多只会返回一条符合条件的记录.这可能是在 const 之外最好的联接类型了,简单的 select 查询不会出现这种 type.
- ref
使用非唯一索引或非唯一索引前缀进行的查找;
相比 eq_ref,不使用唯一索引,而是使用普通索引或者唯一性索引的部分前缀,索引要和某个值相比较,可能会找到多个符合条件的行.
- range
按指定范围(如in()、between and、>、<、>=等,But the premise is that this field needs to be indexed)来检索,很常见.
如:select * from student where id < 5,idThere must be an index on it.
- index
全"表"扫描,But it scans the index tree,通常比 ALL 快,因为索引文件通常比数据文件小,index Scanning is done by scanning through a binary tree,而 ALL is to scan the physical table.也就是说虽然 ALL 和 index 都是读全表,但 index 是从索引中读取的,而 ALL 是从硬盘中读的.
例如:select name from student,但 name The field needs to be indexed,That is, the query field belongs to the field in the index.
- ALL
全表扫描,Scan the complete physical table,This is where optimization is needed.
3.6 possible_keys
指出 MySQL Possible indexes that can be used on this table,Displays the name of the index,Multiple indices are separated by commas.
explain 时可能出现 possible_keys 有列,而 key 显示 NULL 的情况,这种情况是因为表中数据不多,mysql认为索引对此查询帮助不大,选择了全表查询.
如果该列是NULL,则没有相关的索引.在这种情况下,可以通过检查 where 子句看是否可以创造一个适当的索引来提高查询性能,然后用 explain 查看效果.
3.7 key
MySQLDecide which index to actually use,Displays the name of the index,Multiple indices are separated by commas.
如果没有使用索引,则该列是 NULL.如果想强制 MySQL 使用或忽视 possible_keys 列中的索引,在查询中使用 force index、ignore index.
3.8 key_len
表示索引中使用的字节数,可通过该列计算查询中使用的索引的长度(key_len显示的值为索引字段的最大可能长度,并非实际使用长度,即key_len是根据表定义计算而得,不是通过表内检索出的).
不损失精确性的情况下,长度越短越好.
key_len只计算where条件用到的索引长度,而排序和分组就算用到了索引,也不会计算到key_len中.
举例来说,film_actor 的联合索引 idx_film_actor_id 由 film_id 和 actor_id 两个 int 列组成,并且每个 int 是4字节.通过结果中的 key_len = 4 可推断出查询使用了第一个列:film_id 列来执行索引查找.
计算key_len的公式:
varchr(10)变长字段且允许NULL = 10 * ( character set:utf8=3, gbk=2, latin1=1) + 1(NULL) + 2(变长字段)
varchr(10)变长字段且不允许NULL = 10 *( character set:utf8=3, gbk=2, latin1=1) + 2(变长字段)
char(10)固定字段且允许NULL = 10 * ( character set:utf8=3, gbk=2, latin1=1)+1(NULL)
char(10)固定字段且不允许NULL = 10 * ( character set:utf8=3, gbk=2, latin1=1)
3.9 ref
列与索引的比较,表示上述表的连接匹配条件,即哪些列或常量被用于查找索引列上的值,常见的有:const(常量),字段名(例:film.id)
3.10 rows
显示 MySQL The number of rows that are considered necessary to return when executing the query,可结合 type 和 key 分析,without using the index,会全表扫描.rows 的值越小越好,Indicates that the retrieved data is small,注意这个不是结果集里的行数.
3.11 filtered
给出了一个百分比的值,This percentage value and rows 列的值一起使用,Those that will be and the previous table in the execution plan can be estimated(前一个表就是指 id 列的值比当前表的 id small table)The number of rows to join.
3.12 Extra
This field displays some additional information,But some values of this field have optimized reference meaning:
- using where
查询的列未被索引覆盖,where 筛选条件非索引的前导列.
- using index:
表示使用了覆盖索引.这个值重点强调了只需要使用索引就可以满足查询表的要求,No direct access to table data is required.
查询的列被索引覆盖,并且 where 筛选条件是索引的前导列, 是性能高的表现.一般是使用了覆盖索引(索引包含了所有查询的字段).对于innodb来说,If it is secondary index performance There will be many improvements.
- NULL
查询的列未被索引覆盖,并且 where 筛选条件是索引的前导列,意味着用到了索引,但是部分字段未被索引覆盖,必须通过"回表"来实现,不是纯粹地用到了索引,也不是完全没用到索引.
- using join buffer
This value emphasizes that no index is used when getting the join condition,并且需要连接缓冲区来存储中间结果.如果出现了这个值,那应该注意,根据查询的具体情况可能需要添加索引来改进性能
- using filesort
这是 order by 语句的结果.这可能是一个CPU密集型的过程.using filesort Indicates that in-file sorting occurs,MySQL 会对结果使用一个外部索引排序,而不是按索引次序从表里读取行.此时 MySQL 会根据联接类型浏览所有符合条件的记录,并保存排序关键字和行指针,然后排序关键字并按顺序检索行信息.Indicates a very bad phenomenon,必须要优化,特别是大表,可以通过选择合适的索引来改进性能,用索引来为查询结果排序.
- using temporary:
MySQL A temporary table needs to be created to hold the intermediate results. 也就是说,The data needs to be put into a temporary table first,Then get the required data from the temporary table.This temporary table appears,It is also a place that must be optimized,Especially when the amount of data is large.Two common reasons are used on columns from different tables distinct,或者使用了不同的 order by 和 group by 列.
首先是想到用索引来优化.
4. 总结
- EXPLAIN 不会告诉你关于触发器、存储过程的信息或用户自定义函数对查询的影响情况;
- EXPLAIN 不考虑各种 Cache;
- EXPLAIN 不能显示 MySQL 在执行查询时所作的优化工作;
- 部分统计信息是估算的,并非精确值;
- EXPALIN 只能解释 SELECT 操作,其他操作要重写为 SELECT 后查看执行计划.
边栏推荐
猜你喜欢
The use of DDR3 (Naive) in Xilinx VIVADO (2) Read and write design
Hands-on Deep Learning_LeNet
微信公众号之底部菜单
知道创宇EDR系统实力通过中国信通院端点检测与响应产品能力评测
手搓一个“七夕限定”,用3D Engine 5分钟实现烟花绽放效果
TensorFlow学习记录(三):高阶操作 & 神经网络与全连接层
第10章 模块和包
涨姿势了!原来这才是多线程正确实现方式
【HMS core】【FAQ】Account Kit、MDM能力、push Kit典型问题合集6
多行函数;group_by分组;having分组后筛选;单表查询总结
随机推荐
蒲丰投针学习笔记
IBM Q复制新增QSUB
Redis (1) installation and configuration
免费翻译软件哪个好用
到底什么是JS原型
LeetCode每日一题(858. Mirror Reflection)
ESP8266-Arduino编程实例-APDS-9930环境光和趋近感器驱动
深度学习------pytorch实现划拳模型训练
244 page PDF!"2022 China cloud computing ecological blue book published
Apache Doris 1.1 特性揭秘:Flink 实时写入如何兼顾高吞吐和低延时
Flutter 使用 json_serializable 解析 JSON 支持泛型
matlab串口读写
独立站卖家如何使用 WhatsApp Business API 建立有意义的客户关系?
【RISC-V】Trap和Exception
200PLC转以太网与研华webaccess modbusTCP客户端在空调机上应用配置案例
企业应当实施的5个云安全管理策略
yolov5——detect.py代码【注释、详解、使用教程】
项目管理前景
树莓派入门
小程序实战(三) - head组件的封装与使用