MySQL中explain的用法

news2024/11/16 1:44:29

执行结果各字段的含义

EXPLAIN + SQL语句
如:

EXPLAIN SELECT * FROM test

执行结果:

列名描述
id在一个大的查询语句中每个SELECT关键字都对应一个 唯一的id
select_typeSELECT关键字对应的那个查询的类型
table表名
partitions匹配的分区信息
type针对单表的访问方法(重要)
possible_keys可能用到的索引
key实际上使用的索引
key_len实际使用到的索引长度
ref当使用索引列等值查询时,与索引列进行等值匹配的对象信息
rows预估的需要读取的记录条数
filtered某个表经过搜索条件过滤后剩余记录条数的百分比
Extra一些额外的信息

EXPLAIN各列作用

为了让大家有比较好的体验,我们调整了下 EXPLAIN 输出列的顺序。

1. table

表名不论我们的查询语句有多复杂,里面包含了多少个表,到最后也是需要对每个表进行单表访问的,所以MySQL规定EXPLAIN语句输出的每条记录都对应着某个单表的访问方法,该条记录的table列代表着该表的表名(有时不是真实的表名字,可能是简称)

#1. table:表名
#查询的每一行记录都对应着一个单表
explain select count(*) from s1;

image-20220326120805996

这里只用到s1这一张表,所以结果只有一条数据。

#s1:驱动表  s2:被驱动表
EXPLAIN SELECT * FROM s1 INNER JOIN s2;
# 驱动表和被驱动表是 优化器决定的,他认为哪个比较好久用哪个

由于驱动表和被驱动表是由优化器决定的,因此就像这里:s1 INNER JOIN s2 。不一定是s1就是驱动表,结果中s1不一定总在s2上面!

image-20220326121611806

这里用了内连接,涉及两张表,结果就有两条数据。

用到多少个表,就会有多少条记录(临时表也算在记录里面)

2.id

正常来说一个select 一个id ,也有例外的可能,查询优化器做了优化

1.mysql> EXPLAIN SELECT * FROM s1 WHERE key1 = 'a';

image-20220326122616487

这里只有一个select,所以结果也只有一种id

2.mysql> EXPLAIN SELECT * FROM s1 INNER JOIN s2;

image-20220326122717663

这里虽然有两张表,但只有一个select语句,因此执行结果中只有一种id。

3.mysql> EXPLAIN SELECT * FROM s1 WHERE key1 IN (SELECT key1 FROM s2) OR key3 = 'a';

image-20220326122751920

这里出现了子查询,有两个select,因此有两种id。先执行id越大(越接近1越小),越先执行。这里就先执行id为2的查询。

几种特殊情况

1.查询优化器优化

 ######查询优化器可能对涉及子查询的查询语句进行重写,转变为多表查询的操作########
 EXPLAIN SELECT * FROM s1 WHERE key1 IN (SELECT key2 FROM s2 WHERE common_field = 'a');

运行结果: id 只有一种,原因是查询优化器做了优化,变为了多表查询

image-20220326122857145

2.Union去重

原本想的1个select 一个 id , 预计两个。

 #Union去重
# union 去重,union all 不去重
EXPLAIN SELECT * FROM s1 UNION SELECT * FROM s2;

image-20220326123056983

注:这里用的是union,会把s1和s2取交集的集合做成临时表,在这个临时表里面去重。因此有三条记录
可以看到第三条数据的Extra字段表示:Using temporary。说明当前表是临时表(一条记录就是一张表)
因为前面说了,多少种id对应多少个select语句。但这里只有两个select,所以第三条语句的id为NULL。

# union all 不去重  所以不需要放在临时表里面
mysql> EXPLAIN SELECT * FROM s1 UNION ALL SELECT * FROM s2;

image-20220326123147690

不去重,所以不用建立临时表,因此结果只有两条数据(两张表)

小结:

  • id如果相同,可以认为是一组,从上往下顺序执行
  • 在所有组中,id值越大,优先级越高,越先执行
  • 关注点:id号每个号码,表示一趟独立的查询, 一个sql的查询趟数越少越好。因为id种类越多,表示嵌套越多。三层嵌套,三种id,就像Java中的时间复杂度O(n³)一样。而多表连接查询的方式就相当于加法只会有三个一样的id,一种id,三张表连接就像时间复杂度为O(x+y+z)一样,效率高很多。

3.select_type

        一条大的查询语句里边可以包含若干个SELECT关键字,每个SELECT关键字代表着一个小的查询语句,而每个SELECT关键字的FROM子句中都可以包含若干张表(这些表用来做连接查询),每一张表都对应着执行计划输出中的一条记录,对于在同一个SELECT关键字中的表来说,它们的id值是相同的。

MySQL为每一个SELECT关键字代表的小查询都定义了一个称之为select_type的属性,意思是我们只要知道了某个小查询的select_type属性,就知道了这个小查询在整个大查询中扮演了一个什么角色,我们看一下 select_type都能取哪些值,请看官方文档:

名称描述
SIMPLESimple SELECT (not using UNION or subqueries)
PRIMARYOutermost SELECT
UNIONSecond or later SELECT statement in a UNION
UNION RESULTResult of a UNION
SUBQUERYFirst SELECT in subquery
DEPENDENT SUBQUERYFirst SELECT in subquery, dependent on outer query
DEPENDENT UNIONSecond or later SELECT statement in a UNION, dependent on outer query
DERIVEDDerived table
MATERIALIZEDMaterialized subquery
UNCACHEABLE SUBQUERYA subquery for which the result cannot be cached and must be re-evaluated for each row of the outer query
UNCACHEABLE UNIONThe second or later select in a UNION that belongs to an uncacheable subquery (see UNCACHEABLE SUBQUERY)
  • SIMPLE

     # 查询语句中不包含`UNION`或者子查询的查询都算作是`SIMPLE`类型
     EXPLAIN SELECT * FROM s1;
     
      #连接查询也算是`SIMPLE`类型
     EXPLAIN SELECT * FROM s1 INNER JOIN s2;
  • PRIMARY 与 UNION与 UNION RESULT

    • UNION RESULT

      MySQL选择使用临时表来完成UNION查询的去重工作,针对该临时表的查询的select_type就是UNION RESULT,例子上边有。

    #对于包含`UNION`或者`UNION ALL`或者子查询的大查询来说,它是由几个小查询组成的,其中最左边的那个
    #查询的`select_type`值就是`PRIMARY`
     
    #对于包含`UNION`或者`UNION ALL`的大查询来说,它是由几个小查询组成的,其中除了最左边的那个小查询
    #以外,其余的小查询的`select_type`值就是`UNION`
    
    #`MySQL`选择使用临时表来完成`UNION`查询的去重工作,针对该临时表的查询的`select_type`就是`UNION RESULT` 	

    测试sql:

     EXPLAIN SELECT * FROM s1 UNION SELECT * FROM s2;	
    

    image-20220326125611904

    说明:这里s1 union s2,s1(由优化器决定,不一定哪张表就在最左边,这里刚好是s1)在最左边,s1表的select_type 为 `PRIMARY`。由于发生了去重,产生临时表<union1,2> 该表的select_type 为
    `UNION RESULT` 。而其他的表s2的`select_type`值就是`UNION`
    
    EXPLAIN SELECT * FROM s1 UNION ALL SELECT * FROM s2;
    

    image-20220326125627303

  • SUBQUERY

           如果包含子查询的查询语句优化器不能够转为对应的semi-join(多表连接)的形式,并且该子查询是不相关子查询,并且查询优化器决定采用将该子查询物化的方案来执行该子查询时,该子查询的第个SELECT 关键字代表的那个查询 的select_type就是 SUBQUERY,比如下边这个查询:

     #子查询:
     #如果包含子查询的查询语句不能够转为对应的`semi-join`的形式,并且该子查询是不相关子查询。
     #该子查询的第一个`SELECT`关键字代表的那个查询的`select_type`就是`SUBQUERY`
     EXPLAIN SELECT * FROM s1 WHERE key1 IN (SELECT key1 FROM s2) OR key3 = 'a';

image-20220326122751920

说明:这里s1的id最小,表示该表的查询在最外面一层,因此其select_type = 'PRIMARY',而s2的子查询是不相关子查询,因此select_type = 'SUBQUERY'。

  • DERIVED

    derived : 衍生,派生

     #对于包含`派生表`的查询,该派生表对应的子查询的`select_type`就是`DERIVED`
     EXPLAIN SELECT * 
     FROM (SELECT key1, COUNT(*) AS c FROM s1 GROUP BY key1) AS derived_s1 WHERE c > 1;
    

    image-20220326141653065

    说明:首先有两个select对应两种id,这里的子查询用到了s1,table就是s1。而子查询的结果作为一张表供外面查询使用。因此外面表的table为derived2(2是子查询的id)。而子查询由于是派生出来作为表供查询的,因此select_type为'DERIVED'。                                                      

  • MATERIALIZED

    materialized: 英 [məˈtɪəri:əˌlaɪzd] 具体化

    #当查询优化器在执行包含子查询的语句时,选择将子查询物化之后与外层查询进行连接查询时,
    #该子查询对应的`select_type`属性就是`MATERIALIZED`
    EXPLAIN SELECT * FROM s1 WHERE key1 IN (SELECT key1 FROM s2); #子查询被转为了物化表 

image-20220326142034981

4.partitions (可略)

  • 代表分区表中的命中情况,非分区表,该项为NULL。一般情况下我们的查询语句的执行计划的partitions列的值都是NULL。
  • https://dev.mysql.com/doc/refman/5.7/en/alter-table-partition-operations.html
  • 如果想详细了解,可以如下方式测试。创建分区表:
-- 创建分区表,
-- 按照id分区,id<100 p0分区,其他p1分区
CREATE TABLE user_partitions (
    id INT auto_increment,
    NAME VARCHAR(12),PRIMARY KEY(id))
    PARTITION BY RANGE(id)(
    PARTITION p0 VALUES less than(100),
    PARTITION p1 VALUES less than MAXVALUE
);
DESC SELECT * FROM user_partitions WHERE id>200;

查询id大于200(200>100,p1分区)的记录,查看执行计划,partitions是p1,符合我们的分区规则

image-20220325201510359

5.type ☆

执行计划的一条记录就代表着MySQL对某个表的执行查询时的访问方法,又称"访问类型”,其中的type列就表明了这个访问方法是啥,是较为重要的一个指标。比如,看到type列的值是ref,表明MySQL即将使用ref访问方法来执行对s1表的查询。

完整的访问方法如下: system , const , eq_ref , ref , fulltext , ref_or_null ,index_merge , unique_subquery , index_subquery , range , index , ALL 。

我们详细解释一下:

  • system

    当表中只有一条记录并且该表使用的存储引擎的统计数据是精确的,比如MyISAM、Memory,那么对该表的访问方法就是system。比方说我们新建一个MyISAM表,并为其插入一条记录:

    mysql> CREATE TABLE t(i int) Engine=MyISAM;
    Query OK, 0 rows affected (0.05 sec)
    
    mysql> INSERT INTO t VALUES(1);
    Query OK, 1 row affected (0.01 sec)

    然后我们看一下查询这个表的执行计划:

    mysql> EXPLAIN SELECT * FROM t;
    +----+-------------+-------+------------+--------+
    | id | select_type | table | partitions | type   |
    +----+-------------+-------+------------+--------+
    |  1 | SIMPLE      | t     | NULL       | system |
    +----+-------------+-------+------------+--------+
    1 row in set, 1 warning (0.00 sec)

    这里如果是 innodb 会变成ALL , 因为innodb系统不会存条数字段。。MyISAM会存储这么一个字段

  • const

     #当我们根据主键或者唯一二级索引列与常数进行等值匹配时,对单表的访问方法就是`const`
     EXPLAIN SELECT * FROM s1 WHERE id = 10005;
     
     EXPLAIN SELECT * FROM s1 WHERE key2 = '10066';

注意:这里id是主键,key2是唯一二级索引。改成key3,由于不唯一,所以type变为All

  • eq_ref

     #在连接查询时,如果被驱动表是通过主键或者唯一二级索引列等值匹配的方式进行访问的
     #(如果该主键或者唯一二级索引是联合索引的话,所有的索引列都必须进行等值比较),则
     #对该被驱动表的访问方法就是`eq_ref`
     EXPLAIN SELECT * FROM s1 INNER JOIN s2 ON s1.id = s2.id;
    
     

    从执行计划的结果中可以看出,MySQL打算将s2作为驱动表,s1作为被驱动表,重点关注s1的访问 方法是 eq_ref ,表明在访问s1表的时候可以 通过主键的等值匹配 来进行访问。

注意:这个s2表的查询先执行,执行完以后s2.id就是一个具体的值。然后再执行s1.id(主键) = 某一个具体的值(s2.id的执行结果) 速度就也挺快的了。

  • ref

     #当通过普通的二级索引列与常量进行等值匹配时来查询某个表,那么对该表的访问方法就可能是`ref`
     EXPLAIN SELECT * FROM s1 WHERE key1 = 'a';

    tips: 类型相同才可以走索引

    EXPLAIN SELECT * FROM s1 WHERE key3 = 10066;
    # 这个是不会走索引的 因为key3 是字符串
    # 类型不一样,mysql会加函数,进行隐式转换,一旦加上函数,就不会走索引了。

    隐式转换以后,索引可能失效!

  • ref_or_null

     #当对普通二级索引进行等值匹配查询,该索引列的值也可以是`NULL`值时,那么对该表的访问方法
     #就可能是`ref_or_null`
     EXPLAIN SELECT * FROM s1 WHERE key1 = 'a' OR key1 IS NULL;

  • index_merge

     #单表访问方法时在某些场景下可以使用`Intersection`、`Union`、
     #`Sort-Union`这三种索引合并的方式来执行查询
     EXPLAIN SELECT * FROM s1 WHERE key1 = 'a' OR key3 = 'a';

    从执行计划的 type 列的值是 index_merge 就可以看出,MySQL 打算使用索引合并的方式来执行 对 s1 表的查询。

  • unique_subquery

     #`unique_subquery`是针对在一些包含`IN`子查询的查询语句中,如果查询优化器决定将`IN`子查询
     #转换为`EXISTS`子查询,而且子查询可以使用到主键进行等值匹配的话,那么该子查询执行计划的`type`
     #列的值就是`unique_subquery`
     EXPLAIN SELECT * FROM s1 
     WHERE key2 IN (SELECT id FROM s2 WHERE s1.key1 = s2.key1) OR key3 = 'a';
  • index_subquery

    EXPLAIN SELECT * FROM s1 WHERE common_field IN (SELECT key3 FROM s2 where
    s1.key1 = s2.key1) OR key3 = 'a';
  • range

    #如果使用索引获取某些`范围区间`的记录,那么就可能使用到`range`访问方法
    EXPLAIN SELECT * FROM s1 WHERE key1 IN ('a', 'b', 'c');
    
    #同上
    EXPLAIN SELECT * FROM s1 WHERE key1 > 'a' AND key1 < 'b';
  • index

    #当我们可以使用索引覆盖,但需要扫描全部的索引记录时,该表的访问方法就是`index`
    EXPLAIN SELECT key_part2 FROM s1 WHERE key_part3 = 'a';

    索引覆盖,INDEX idx_key_part(key_part1, key_part2, key_part3) 这3个构成一个复合索引

    key_part3 在复合索引里面,,查询的字段也在索引里面,干脆就直接遍历索引查出数据

    思考: 好处,索引存的数据少,数据少页就少,这样可以减少io。

  • ALL

mysql> EXPLAIN SELECT * FROM s1;

一般来说,这些访问方法中除了All这个访问方法外,其余的访问方法都能用到索引,除了index_merge访问方法外,其余的访问方法都最多只能用到一个索引。

小结:

结果值从最好到最坏依次是:

system > const > eq_ref > ref >

fulltext > ref_or_null > index_merge >unique_subquery > index_subquery > range >

index > ALL

SQL 性能优化的目标:至少要达到 range 级别,要求是 ref 级别,最好是 consts级别。(阿里巴巴 开发手册要求)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1612450.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

机器学习预测汽车油耗效率 MPG

流程 数据获取导入需要的包引入文件,查看内容划分训练集和测试集调用模型查看准确率 数据获取 链接&#xff1a;https://pan.baidu.com/s/1KeIJykbcVpsfEk0xjhiICA?pwd30oe 提取码&#xff1a;30oe --来自百度网盘超级会员V1的分享导入需要的包 import pandas as pd imp…

【Spring Boot】掌握Spring Boot:深入解析配置文件的使用与管理

&#x1f493; 博客主页&#xff1a;从零开始的-CodeNinja之路 ⏩ 收录文章&#xff1a;【Spring Boot】掌握Spring Boot&#xff1a;深入解析配置文件的使用与管理 &#x1f389;欢迎大家点赞&#x1f44d;评论&#x1f4dd;收藏⭐文章 目录 Spring Boot 配置文件一. 配置文…

重新总结一下以前写过的“波特率”!单片机常见的通信速率分析!

文章目录 如题以前文章新的总结如题 波特率是单片机中描述通信速率的一个单位,比如串口通信、SPI通信、IIC通信、LIN通信、CAN通信等等,现在重新总结一下涉及到波特率的一些知识点。 以前文章 上面是存储的单位换算方式 这是通信速率的换算方式 新的总结 波特率的英文是…

画家-qt-surce

void GraphicView::paintEvent(QPaintEvent *pe) { QPainter painter(viewport()); painter.setRenderHint(QPainter::SmoothPixmapTransform);//升级画家 painter.drawImage(rect(),musicImage); } 分析&#xff1a; 这段代码是用于绘制图形视图的部分。 1. void GraphicV…

JavaCard学习笔记: CAP Component 之 Class Component

文章目录 整体结构tag和size字段signature_pool_length和signature_pooltype_descriptor结构导入类型编码导入项签名示例导入类导入数组导入远程方法 interfaces[]interface_info结构flagsinteface_countsuperinterfacesinterface_name class_info_compact classes[]结构flagsi…

mapreduce中的ReduceTask工作机制(Hadoop)

ReduceTask 是 Hadoop 中的一个重要组件&#xff0c;负责对 MapTask 的输出进行合并、排序和归并&#xff0c;最终生成最终的输出结果。 ReduceTask 的工作机制 1. 分组&#xff08;Shuffle&#xff09;阶段&#xff1a; 在分组阶段&#xff0c;ReduceTask 会从多个 Mapper …

【问题处理】银河麒麟操作系统实例分享,服务器操作系统VNC远程问题分析

1.服务器环境以及配置 【内核版本】 4.19.90-23.8.v2101.ky10.aarch64 【OS镜像版本】 0518-server 2.问题现象描述 服务器通过vncserver:1.service服务启动的vnc服务后&#xff0c;普通用户用vnc连接时&#xff0c;锁屏后&#xff0c;然后输入登陆密码会报密码错误&…

回溯算法练习day.4

93.复原ip地址 链接&#xff1a;. - 力扣&#xff08;LeetCode&#xff09; 题目描述&#xff1a; 有效 IP 地址 正好由四个整数&#xff08;每个整数位于 0 到 255 之间组成&#xff0c;且不能含有前导 0&#xff09;&#xff0c;整数之间用 . 分隔。 例如&#xff1a;"…

基于单片机的智能病床呼叫系统设计与仿真

摘 要 本文设计的病床呼叫系统采用单片机作为控制器。该系统具有远程控制、病人的身体情况检测、报警呼叫、显示和执行器运动的功能。远程控制由红外线传感器和矩阵键盘组成&#xff0c;检测电路由温湿度传感器DH22、心率传感器Pulse Sensor、压力传感器MPX4115组成&#x…

苹果电脑虚拟机黑屏了怎么办解决 MAC系统升级后虚拟机黑屏问题 苹果电脑虚拟机卡住了怎么办

虚拟机是一种可以在一台电脑上运行多个操作系统的软件&#xff0c;它可以让用户在苹果电脑上安装和使用Windows、Linux等其他系统。但是&#xff0c;有时候在升级Mac系统或者虚拟机软件后&#xff0c;虚拟机会出现黑屏的现象&#xff0c;无法正常启动或者使用。这种情况该如何解…

java swing电商出入库管理系统eclipse开发Mysql数据库CS结构java编程

一、源码特点 java swing 电商出入库管理系统 是一套完善的窗体设计系统&#xff0c;对理解SWING java 编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;&#xff0c;系统主要采用C/S模式开发。 应用技术&#xff1a;javamysql 开发工具&#xff1…

一堆喷儿香喷儿香的工具网站-已经收藏-搜嗖工具箱!

文心一言 https://yiyan.baidu.com/ ​ ChatGpt横空出世的横空出世好像一把钥匙&#xff0c;开启了大模型时代&#xff0c;国内也有不错的产品&#xff0c;比如百度的文心一言&#xff0c;从3.5到4.0看得见的成长&#xff0c;现在的文心一言是我们工作中不可缺少的好帮手&am…

vulfocus靶场之redis命令执行cve-2022-0543漏洞复现

漏洞&#xff1a; Redis是著名的开源Key-Value数据库&#xff0c;其具备在沙箱中执行Lua脚本的能力。 Debian以及Ubuntu发行版的源在打包Redis时&#xff0c;不慎在Lua沙箱中遗留了一个对象package&#xff0c;攻击者可以利用这个对象提供的方法加载动态链接库liblua里的函数&…

【Visual Studio 2012中文版】下载安装以及使用方法

文章目录 前言一、下载安装包二、安装步骤1.双击VS2012_ULT_chs.iso文件打开2.双击vs_ultimate.exe打开安装程序3.选择要安装的功能4.软件正在安装&#xff0c;请耐心等待10分钟5.安装成功&#xff0c;点击“启动”6.激活码&#xff08;产品密钥&#xff09; 三、VS2012使用&am…

【Java】实现一个简单的线程池

&#x1f4dd;个人主页&#xff1a;哈__ 期待您的关注 一、线程池的模式 线程池顾名思义就是管理线程的一个池子&#xff0c;我们把创建线程的过程交给线程池来处理&#xff0c;而这个线程池当中的线程都会从阻塞队列当中取获取任务执行。 我们不在直接把任务的创建过程写到…

京东微服务microApp使用总结

前言 基于现有业务门户进行微服务基础平台搭建 主应用框架&#xff1a;vue3vite 子应用框架&#xff1a;vue2webpack / vue3vite在这里插入代码片 本地调试即可&#xff1a;主应用子应用进行打通&#xff08;注意&#xff1a;两者都是vue3vite&#xff09; 问题总结 1.嵌入…

基于SSM+Jsp+Mysql的高校毕业生就业满意度调查统计系统

开发语言&#xff1a;Java框架&#xff1a;ssm技术&#xff1a;JSPJDK版本&#xff1a;JDK1.8服务器&#xff1a;tomcat7数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09;数据库工具&#xff1a;Navicat11开发软件&#xff1a;eclipse/myeclipse/ideaMaven包…

new[]与delete[]

&#xff08;要理解之前关于new,delete的一些概念&#xff0c;看​​​​​​ CSDN&#xff09; 引子&#xff1a; 相比new&#xff0c;new[]不仅仅是个数的增加&#xff0c;还有int大小记录空间的创建&#xff0c; 下图中错误的用模拟多个new来替代new[]&#xff0c;释放步…

从C到JAVA之学习JAVA的第一周笔记

文章目录 java语言概述JDK与JRE编写执行过程第一份java代码解读编写编译运行其他 注释三种注释方法 java API文档关键字标识符数据类型基本数据类型自动类型提升规则引用数据类型 string概述String与基本数据类型的变量间的运算 运算符键盘录入运行控制语句数组定义与静态初始化…

【MyBatis】(MyBatis 其他查询操作 多表查询 #{} 和 ${} 排序功能 like 查询 #{} 和 ${} 的区别 数据库连接池)

文章目录 MyBatis其他查询操作多表查询#{} 和 ${}排序功能like 查询#{} 和 ${} 的区别 数据库连接池 MyBatis 其他查询操作 创建表&#xff1a; -- 创建⽂章表 DROP TABLE IF EXISTS articleinfo; CREATE TABLE articleinfo (id INT PRIMARY KEY auto_increment,title VARCH…