MySQL的Explain总结

Explain简介

MySQL优化器在基于成本的计算和基于规则的SQL优化会生成一个所谓的 执行计划,我们就可以使用执行计划查看MySQL对该语句具体的执行方式。

介绍这个好啰嗦就是了,我们可以通过这个优化器展示的执行计划,查看优化器对我们的SQL进行优化的步骤,连接转换成单表访问时的优化。以及对于之前知识的复习了属于是,比如访问方式,索引的选择,半连接等SQL语句优化。

mysql> explain select * from mall.ums_admin;
+----+-------------+-----------+------------+------+---------------+------+---------+------+------+----------+-------+
| id | select_type | table     | partitions | type | possible_keys | key  | key_len | ref  | rows | filtered | Extra |
+----+-------------+-----------+------------+------+---------------+------+---------+------+------+----------+-------+
|  1 | SIMPLE      | ums_admin | NULL       | ALL  | NULL          | NULL | NULL    | NULL |    5 |   100.00 | NULL  |
+----+-------------+-----------+------------+------+---------------+------+---------+------+------+----------+-------+
1 row in set, 1 warning (0.00 sec)

上述是我们使用explain使用的简单小例子。下面是每列的简单描述。

MySQL的Explain总结

我们接下来就是对于Explain中展示的列进行介绍,回涉及到之前的SQL成本和语句优化。

table列

对于每个SQL语句来说,不管是不是使用连接的多表查询,我们最后都会变成执行多个单表查询的语句。所以table列就指明了当前执行某个计划的表是哪一个。就比如上面例子的单表查询的SQL。

ID列

会有以下几种情况

  1. 单表查询,此时的ID列就为1。
  2. 连接查询,此时对于两个表的连接查询ID列都为1,MySQL视在前面的表就是作为驱动表,在后面的表就是作为被驱动表。

MySQL的Explain总结
  1. 子查询,有多种情况
    MySQL的Explain总结
    MySQL的Explain总结
  2. UNION连表,就是出现一个select就再分配一个ID值。稍微有点特殊对于UNION ALL。

MySQL的Explain总结

​ 首先呢是因为UNION关键字会对合并的结果集进行去重,进行去重就得创建一个临时表来进行去重,第三行是一个临时表。但是呢UNION ALL不去重就不会出现第三行记录。

MySQL的Explain总结

select_type列

MySQL的Explain总结

Simple

它是一个简单的单表查询或联接查询。

[En]

It is a simple single table query or join query.

PRIMARY

就是UNION或UNION ALL或子查询的情况下,最左边的select就是PRIMARY类型的。

MySQL的Explain总结

UNION

就是UNION的时候除了最左边的SELECT是PRIMARY类型,其他的SELECT都是UNION。

UNION RESULT

UNION的结果就会创建一个临时表进行去重,临时表就是这个select_type。

SUBQUERY

  1. 子查询不能转换为半连接
  2. 该子查询是 不相关子查询
  3. 子查询就会考虑使用物化的方案(如果数量很少就不会物化)

以上三个条件成立,此时子查询的第一个select部分的select_type就是SUBQUERY。如果物化的话,子查询只会执行一次,应该不用多说了。

DEPENDENT SUBQUERY

  1. 子查询不能转换为半连接
  2. 子查询是相关子查询

以上条件成立,此时子查询的第一个select部分的select_type就是DEPENDENT SUBQUERY。相关外层查询会不断传参然后一直进行子查询。

DEPENDENT UNION

UNION中除了最左边的select部分,被union连接的小查询的select_type 都是DEPENDENT UNION。

DERIVED

采用物化的方式执行的派生表,其中的子查询的select_type就是DERIVED。

MySQL的Explain总结

MATERIALIZED

当子查询是物化后转连接的方式,就是首先子查询是不相关子查询,然后子查询执行物化,优化成连接的方式和外层查询。此时子查询的select_type就是MATERIALIZED。

MySQL的Explain总结

partitions列

一般情况下都为null。

type列

介绍如何访问访问当前行的表。

[En]

Is to introduce how to access the table that accesses the current row.

我们之前学习的有

  • const 主键或唯一索引等值查询
  • ref 二级索引等值查询
  • ref_or_null 二级索引等值查询或null值
  • range 范围查询
  • index 使用二级索引进行结果覆盖
  • all 全表扫描

新的

  • system 当表中只有一条记录,且该表的存储引擎的统计数据是精确的(MYISAM、MEMORY),此时访问就是system。
  • eq_ref 连接查询时,对于被驱动表是使用主键或唯一索引进行访问的,被驱动表的访问方式就是eq_ref。
  • index_merge 索引合并,就是intersection索引交集或者union索引并集,sort_union排序并集。
  • unique_subquery 就是优化器将外层查询的in优化为exists,子查询可以使用主键或唯一索引进行查询,此时子查询就是unique_subquery 。???
  • index_subquery 就是上面的变种,此时子查询使用普通的二级索引就是这个访问方式。

possible_key和key列

possible_key就是SQL可以用到的索引。

key就是优化器计算成本后决定使用的索引列。

key_len列

使用的索引的索引记录的长度。有三个部分加在一起。

[En]

Is the length of the index record of the index used. There are three parts added together.

  • 如果是固定长度的,int这类的就是其的固定长度4。如果是变长的,比如varchar(100)且字符集utf8的就是最大长度300。
  • 能不能为null值,如果可以需要1字节。
  • 记录可变长度字段默认长度为2字节,需要增加可变长度字段(也需要增加字符集长度)。
    [En]

    the length of the record variable length field is 2 bytes by default, and the variable length field needs to be added (character set length also needs to be added).*

对于上述三个部分,前面已经很清楚了。

[En]

For the above three parts, the previous has been very clear.

如果是varchar(100)可以为null的列,就需要300字节数据最大长度+1字节的非null+2字节记录长度,总长度就是303字节。

对于联合索引来说,用几个索引这个值就根据索引进行叠加上去,606就是俩varchar(100)可以为null的字段的联合索引。

ref列

也就是说,当我们使用索引进行匹配时,特定于索引列的等价物匹配什么、常量值或列等等。

[En]

That is, when we use the index to match, the index column specific equivalent matches what, the constant value or a column, and so on.

MySQL的Explain总结

rows列

表示优化程序预计要扫描表或索引的行数。

[En]

Represents how many rows the optimizer predicts to scan the table or index.

filter列

表示驱动表的扇出率,例如驱动表有自己的查询条件后,预计有多少条记录满足条件,并输出。

[En]

Represents the fan-out ratio of the driver table, such as how many records are expected to meet the conditions after the drive table has its own query conditions, and output them.

此字段对于单个表没有意义,但对于联接查询,您可以预期受驱动表将被执行的次数,即受驱动表的扇出值。

[En]

This field makes no sense for a single table, but for join queries, you can expect how many times the driven table will be executed, that is, the fan out value of the driven table.

extra列

正是优化器为我们提供了帮助我们分析语句的附加信息。

[En]

It is the optimizer that provides us with additional information to help us analyze statements.

  • No tables used
  • 表示查询语句没有from,没有从表被使用到。
  • Impossible where
  • 顾名思义,不可能满足的where语句,如1 != 1这种永远不可能满足
  • No matching min/max row
  • 使用min和max聚集函数但是where语句没有能匹配的记录。
  • Using index
  • 就是索引覆盖嘛。忘了可以回前面SQL优化文章看看
  • Using index condition
  • 就是索引一个条件能使用,一个条件不能使用,导致我们使用索引先进行一个条件的查询,然后再对查询结果进行另外一个条件的过滤,然后再回表。
  • Using where
  • 使用全表扫描或索引扫描,然后语句中有where的普通条件需要一条一条进行过滤,通常是没有使用索引的列,就会出现Using where的备注。
  • Using join buffer
  • 我们使用join buffer加快驱动表对于被驱动表的连接。
  • Not exists
  • 当进行外连接的时候,被驱动表的搜索条件为某个列 is null 但是列本身是not null的,这时就会出现not exists
  • Using intersect(…) ,Using union(…),Using sort_union(…)
  • 当出现着这个备注就表示使用了索引合并。
  • Zero limit
  • 出现limit 0,这种不要结果的SQL。
  • Using filesort
  • 当无法使用索引排序,就会将数据在内存(数据量少的时候)或者在磁盘(数据量多的时候)进行对应条件的排序。这是十分耗时的,这是能用索引尽量用索引。
  • Using temporary
  • 使用了临时表
  • Start temporary , End temporary
  • 就是查询优化器使用半连接优化时,使用的是临时表对连接结果去重的方案实现半连接的时候,驱动表就会在extra中显示Start temporary,而被驱动表显示End temporary。
  • LooseScan
  • 就是我们使用半连接的LooseScan的策略实现时,就会在驱动表中出现这个。
  • FirstMatch
  • 忘了去SQL优化看了一下,就是最笨的那个驱动表一条一条匹配被驱动表的方式,就会在被驱动表显示此注释。

Original: https://www.cnblogs.com/duizhangz/p/16315184.html
Author: 大队长11
Title: MySQL的Explain总结

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/505344/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • canal

    canal 简介 canal 主要用途是基于 MySQL 数据库增量日志解析,提供增量数据订阅和消费。 canal 工作原理: canal 模拟 MySQL slave 的交互协议…

    数据库 2023年6月14日
    0127
  • Figma 快捷键

    作用 WINDOWS MAC 窗口切换到Home Ctrl + 1 Cmd + 1 窗口切换到打开的第一个文件 Ctrl + 2 Cmd + 2 打开菜单搜索 Ctrl + / C…

    数据库 2023年6月6日
    0133
  • Linux中MySQL的安装以及卸载

    一.MySQL MySQL 是一种开放源代码的关系型数据库管理系统,开发者为瑞典MySQL AB公司。在2008年1月16号被Sun公司收购。而2009年,SUN又被Oracle收…

    数据库 2023年5月24日
    0155
  • Django设置跨域访问

    Django设置跨域访问 pip install django-cors-headers (2) settings.py 配置如下 INSTALLED_APPS = [ # ‘dj…

    数据库 2023年6月14日
    0138
  • docker安装下载

    一、docker安装 查看 sudo systemctl status docker 安装 …

    数据库 2023年6月9日
    0129
  • Centos8 python pip报错: pip is configured with locations that require TLS/SSL 解决

    在使用pip进行软件包安装的时候出现问题: WARNING: pip is configured with locations that require TLS/SSL, howe…

    数据库 2023年6月11日
    0126
  • 数据库持久化+JDBC数据库连接

    数据持久化就是 将内存中的数据模型转换为存储模型,以及 将存储模型转换为内存中的数据模型的统称。数据模型可以是任何数据结构或对象模型,存储模型可以是关系模型、XML、二进制流等。 …

    数据库 2023年6月16日
    0191
  • Flyweight 享元(结构型)

    一:描述:(该模式实际应用较少) Flyweight 享元模式是对大量细粒度的元素进行共享和重用。减少对象的创建减轻内存; 注和单例模式不同的是:享元模式的各个对象佣有各自的行为并…

    数据库 2023年6月11日
    0165
  • MySQL提示sql_mode=only_full_group_by解决办法

    MySQL异常sql_mode=only_full_group_by 原因:在MySQL 5.7后MySQL默认开启了SQL_MODE严格模式,对数据进行严格校验。会报sql_mo…

    数据库 2023年6月14日
    0136
  • SQL的约束

    概念:约束是作用于表中的字段以限制表中数据存储的规则 [En] concept: constraints are rules that act on fields in a tab…

    数据库 2023年5月24日
    0158
  • form表单上传文件及后端获取文件

    写在前面(强调) 使用form表单上传文件时必须要干的两件事: 1.form表单的method&#x630…

    数据库 2023年6月14日
    0178
  • 系统发布springboot项目

    首先把项目终止服务,但这样的做法不是很好,也可以来个配置文件 1.查看运行的java文件有哪些,位置在哪里,端口号是多少 ps -ef | grep java 2.结束端口进程的运…

    数据库 2023年6月6日
    0131
  • JavaWeb 05_JDBC入门及连接MySQL

    一、概念 *概念: Java DataBase Connectivity Java数据库连接, Java语言操作数据库* JDBC本质:其实是官方(sun公司)定义的一套操作所有关…

    数据库 2023年5月24日
    0161
  • [Npoi]Npoi导入Excel, 转为Entity

    Npoi导入Excel其实只要读成DataTable就可以随意操作了, 比如转为Entity… By: 胖纸不争NetCore🐧群: 743336452 核心代码: p…

    数据库 2023年6月9日
    0116
  • Linux–>网络配置

    虚拟机NAT网络关系图 在Linux中查看网络配置 ifconfig ping 测试主机之间网络连通性 测试当前服务器是否可以连接目的主机 ping 目&am…

    数据库 2023年6月14日
    0140
  • Linux网络配置

    Linux网络配置 NAT网络配置 查看网络IP和网关 可以在 编辑->虚拟网络编辑器中 查看网络IP和网关 说明:1.什么是IP协议/地址?即”网络之间能相互连…

    数据库 2023年6月16日
    0142
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球