听说你对explain 很懂?

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: explain所有人都应该很熟悉,通过它我们可以知道SQL是如何执行的,虽然不是100%管用,但是至少大多数场景通过explain的输出结果我们能直观的看到执行计划的相关信息。

explain所有人都应该很熟悉,通过它我们可以知道SQL是如何执行的,虽然不是100%管用,但是至少大多数场景通过explain的输出结果我们能直观的看到执行计划的相关信息。

早一些的版本explain还只能查看select语句,现在已经能支持deleteupdateinsertreplace了。

刚开始我想写这个的时候只是因为这个东西经常性不用就忘记,写了发现其实这个东西真的挺麻烦的,要把每个场景都整出来麻烦的很。

id

查询编号,如果没有子查询或者联合查询的话,就只有一条,如果是联合查询的话,那么会出现一条id为null的记录,并且标志查询结果,因为union结果会放到临时表中,所以我们看到这里的表名是<union1,2>这种格式。

079a2a7cd93cfaad61b9fab96a5b5cf7.jpg

select_type

关联类型,决定访问表的方式。

9c86883437246b722e7fb7c49960edab.jpg

SIMPLE

简单查询,代表没有子查询或者union

PRIMARY

如果不是简单查询,那么最外层查询就会被标记成PRIMARY。

UNION&UNION RESULT

从上图可以看出来了,包含联合查询,第一个被标记成了PRIMARY,union之后的查询被标记成UNION,以及最后产生的UNION RESULT

DERIVED

用来标记出现在from里的子查询,这个结果会放入临时表中,也叫做派生表。

3b5e772df756f6b5f1c478327010f4b0.jpg

这个对于低版本的Mysql可能显示是这样的,高一点可能你看到的还是PRIMARY,因为被Mysql优化了。我换一个版本的Mysql和SQL执行可以验证到这个结果。

d639217fa3da60cbd43666ba33ac156f.jpg

SUBQUERY

不在from里的子查询。

d02c2971acc8b93f901de6fcb26583ce.jpg

DEPENDENT

代表关联子查询(子查询使用了外部查询包含的列),和UNIONSUBQUERY组合产生不同的结果。

9782879eaa2b58a918b705570acb2221.jpg

UNCACHEABLE

代表不能缓存的子查询,也可以和UNIONSUBQUERY组合产生不同的结果。

ed0907ca5ea9393630f9ed7c06843c08.jpg

MATERIALIZED

物化子查询是Mysql对子查询的优化,第一次执行子查询时会将结果保存到临时表,物化子查询只需要执行一次。

比如上述DERIVED就是物化的一种体现,与之对应的就是DEPENDENT,每次子查询都需要重新调用。

这个结果无法直观的看出来,可以用FORMAT=JSON命令查看materialized_from_subquery字段。

2300d13eee61aac36d99edde79700f5e.jpg

table

显示表名,从上述的一些图中可以观察到UNION_RESULT和DERIVED显示的表名都有一些自己的命名规则。

比如UNION_RESULT产生的是<unionM,N>,DERIVED产生的是。

partitions

数据的分区信息,没有分区忽略就好了。

type

关联类型,决定通过什么方式找到每一行数据。以下按照速度由快到慢。

system>const>eq_ref>ref>fulltext>ref_or_null>index_merge>unique_subquery>index_subquery>range>index>ALL。

864bfb1d597e770102c89594916bdd8f.jpg

system&const

这通常是最快的查找方式,代表Mysql通过优化最终转换成常量查询,最常规的做法就是直接通过主键或者唯一索引查询。

7b2e4070e7f2ba9352b5ca9711d25a65.jpg

而system是const的一个特例(只有一行数据的系统表),随便找一张系统表,就插入一条数据就可以看到system了。

fdbe20341dc25bd70294375020ab5670.jpg

eq_ref

通常通过主键索引或者唯一索引查询时会看到eq_ref,它最多只返回一条数据。user_id是唯一索引,为了测试就关联以下主键索引。

18818feb08c27bbfcb5db6ebe3f1c76b.jpg

ref

也是通过索引查找,但是和eq_ref不同,ref可能匹配到多条符合条件的数据,比如最左前缀匹配或者不是主键和唯一索引。

最简单的办法,随便查一个普通索引就可以看到。

b957c251e278862cfa81e6ad9fb022c1.jpg

fulltext

使用FULLTEXT索引

ref_or_null

和ref类似,但是还要进行一次查询找到NULL的数据。

这相当于是对于IS NULL查询的优化,如果表数据量太少的话,你或许能看到这里类型是全表扫描。

bc5f6875d75c9f2ff702f1d043786bfa.jpg

index_merge

索引合并是在Mysql5.1之后引入的,就像下面的一个OR查询,按照原来的想法要么用name的索引,要么就是用age的索引,有了索引合并就不一样了。

对于这种单表查询(无法跨表合并)用到了多个索引的情况,每个索引都可能返回一个结果,Mysql会对结果进行取并集、交集,这就是索引合并了。

6b17aed6d31120b89ffce147c1d6c5ba.jpg

unique_subquery

按照官方文档所说,unique_subquery只是eq_ref的一个特例,对于下图中这种in的语句查询会出现以提高查询效率。

由于Mysql会对select进行优化,基本无法出现这个场景,只能用update这种语句了。

5dc0c5c3fa2a703171e05d6f663376c1.jpg

range

看名字就知道,范围查询,其实就是带有限制条件的索引扫描。

常见的范围查询比如between and,>,<,like,in 都有可能出现range。

a71b6ef118e9aa61addd8dcb8beba9c2.jpg

index

跟全表扫描类似,只是扫表是按照索引顺序进行。

ALL

全表扫描,没啥好说的。

possible_keys

可以使用哪些索引。

key

实际决定使用哪个索引。

key_len

索引字段的可能最大长度,不是表中实际数据使用的长度。

ref

表示key展示的索引实际使用的列或者常量。

rows

查询数据需要读取的行数,只是一个预估的数值,但是能很直观的看出SQL的优劣了。

filtered

5.1版本之后新增字段,表示针对符合查询条件的记录数的百分比估算,用rows和filtered相乘可以计算出关联表的行数。

Extra

解析查询的附加额外信息,这个太多了,有兴趣可以自己看官方文档,只列举一些常见的。

Using index

使用覆盖索引。

Using index condition

可以使用索引下推(不一定真的使用了),索引下推简单来说就是加上了条件筛选,减少了回表的操作。

d08134705c5cfe5c6c6d9c9e3c3a151d.jpg

Using temporary

排序使用了临时表。

Using filesort

使用外部索引文件排序,但是不能从这里看出是内存还是磁盘排序,我们只能知道更消耗性能。

Using where

where过滤,没啥好说的。

Zero limit

除非你写个LIMIT 0。

Using sort_union(), Using union(), sing intersect()

使用了索引合并,参看上文。

总结

7cf5f12814fb672a57aa682a1dda5a43.jpg

相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
SQL 关系型数据库 MySQL
Mysql数据库第十三课-----------sql语句的拔高3--------直冲云霄
Mysql数据库第十三课-----------sql语句的拔高3--------直冲云霄
|
24天前
|
SQL 机器学习/深度学习 关系型数据库
最完整的Explain总结,SQL优化不再困难!
最完整的Explain总结,SQL优化不再困难!
|
6月前
|
SQL 关系型数据库 MySQL
零基础快速自学SQL,2天足矣
零基础快速自学SQL,2天足矣
118 0
|
SQL 机器学习/深度学习 关系型数据库
MySql优化神器 Explain工具介绍
MySql优化神器 Explain工具介绍
178 0
|
SQL 关系型数据库 MySQL
MySQL数据库第十四课--------sql优化---------层层递进1
MySQL数据库第十四课--------sql优化---------层层递进
|
SQL 存储 关系型数据库
MySQL数据库第十四课--------sql优化---------层层递进 2
MySQL数据库第十四课--------sql优化---------层层递进
|
存储 机器学习/深度学习 算法
面试之前,MySQL表连接必须过关!——表连接的原理
什么是连接查询?笛卡尔积如何避免?内连接和外连接的概念是什么?表连接的原理是什么?Simple Nested-Loop Join、Index Nested-Loop Join、Block Nested-Loop Join、Hash Join分别是什么概念?怎样分析表连接使用了哪种连接算法?本文带你一探究竟!
148 0
面试之前,MySQL表连接必须过关!——表连接的原理
|
SQL 关系型数据库 MySQL
MySQL数据库第十一课---------SQl语句的拔高-------水平提升
MySQL数据库第十一课---------SQl语句的拔高-------水平提升
|
SQL 存储 机器学习/深度学习
长达1.7万字的explain关键字指南奉上!请你别再说不会SQL优化了
当你的数据里只有几千几万,那么 SQL 优化并不会发挥太大价值,但当你的数据里去到了几百上千万,SQL 优化的价值就体现出来了!因此稍微有些经验的同学都知道,怎么让 MySQL 查询语句又快又好是一件很重要的事情。要让 SQL 又快又好的前提是,我们知道它「病」在哪里,而 explain 关键字就是 MySQL 提供给我们的一把武器!
|
存储 SQL 缓存
1024程序员节|【MySQL从入门到精通】【高级篇】(二十七)外连接和内连接如何进行查询优化呢?join的原理了解一波
【MySQL从入门到精通】【高级篇】(二十六)建了索引就能用么?我看未必。来看看几种索引失效的情况吧 上篇文章我们将来学习索引失效的几种情况。有时候并不是说加了索引,就一定能用上索引,还是要具体情况具体分析。本文将介绍一下MySQL优化器如何对外连接和内连接进行查询优化的以及介绍Join语句的底层原理。
209 0
1024程序员节|【MySQL从入门到精通】【高级篇】(二十七)外连接和内连接如何进行查询优化呢?join的原理了解一波