MySQL Explain 字段详解

Explain 工具介绍

Explain 一般被称为解释器,通过 Explain 工具,我们能分析我们使用的查询语句或是结构的性能瓶颈,它提供 MySQL 如何执行语句的信息。

使用语法:

explain [extended|partition] select

select 关键字前加 explain 关键字,MySQL 会返回该查询的执行计划不是执行这条 SQL。

explain 分析示例

创建表语句

DROP TABLE IF EXISTS `user`;
CREATE TABLE `user`(
`id` INT(11) NOT NULL,
`name` VARCHAR(45) DEFAULT NULL,
`gender` CHAR(1) DEFAULT 0,
PRIMARY KEY(`id`)
) ENGINE=INNODB DEFAULT CHARSET=utf8;INSERT INTO `user` (`id`, `name`,`gender`) VALUES(1,'a',0),(2,'b',1),(3,'c',0);DROP TABLE IF EXISTS `teacher`;
CREATE TABLE `teacher`(
`id` INT(11) NOT NULL,
`name` VARCHAR(45) DEFAULT NULL,
`subject` VARCHAR(20) DEFAULT 0,
PRIMARY KEY(`id`),
KEY `idx_name` (`name`)
) ENGINE=INNODB DEFAULT CHARSET=utf8;INSERT INTO `teacher` (`id`, `name`,`subject`) VALUES(1,'t1',"语文"),(2,'t2',"数学"),(3,'t3',"体育");DROP TABLE IF EXISTS `teacher_user`;
CREATE TABLE `teacher_user`(
`id` INT(11) NOT NULL,
`teacher_id` INT(11) NOT NULL,
`user_id` INT(11) NOT NULL,
PRIMARY KEY(`id`),
KEY `idx_teacher_user_id` (`teacher_id`,`user_id`)
) ENGINE=INNODB DEFAULT CHARSET=utf8;INSERT INTO `teacher_user` (`id`, `teacher_id`,`user_id`) VALUES(1,1,1),(2,3,2),(3,1,2);

使用 explain 工具

EXPLAIN SELECT * FROM USER;

简单使用 explain 工具

Explain 的两个变种

  1. Explain extended:会在explain的基础上额外提供一些查询优化的信息。紧随其后通过show warnings 命令可以得到优化后的查询语句,从而看出优化器优化了什么。额外的还有 filtered 列,它显示的是一个百分比的值,rows * filtered/100 可以估算出将要和 explain 中前一个表进行连接的行数(前一个表指 explain 中的 id 值比当前表 id 值小的表)。
  2. explain partitions:显示查询会访问的分区,输出中增加 partitions 字段。

注意:如果你用的是 MySQL 5.6 ,这两个命令可以用,MySQL 8已经不支持了,使用会报错,不过他们将这两个字段的输出已经整合在默认的 explain 中了,直接后接show warnings就可以看到重构后的命令了。

EXPLAIN SELECT * FROM USER WHERE id=1;
SHOW WARNINGS;

在这里插入图片描述

在这里插入图片描述

Explain 字段详解

含义
idselect 的序列号,有几个 select 就有几个id,id越大,越先执行
select_type表示语句是简单查询还是复杂查询,该列的值有:simple、primairy、subquery、derived、union
table显示 explain 语句正在访问的表
partitions如果查询是基于分区表的话,会显示查询将要访问的分区
type关联的类型,也就是 MySQL 决定如何查找表中的行,查找数据行记录的大概范围
possible_keys可能使用的索引
keyMySQL 实际采用了哪个索引来优化对该表的访问
key_len索引里使用的字节数,通过该列的值可以算出具体使用了索引的哪些列
ref在 key 列记录的索引中,表查找值所用到的列或常量
rowsMySQL预估要读取并检测的行数,不是结果集中的行数
filtered百分比的值,根据 rows*filtered/100 可以估算出将要和 explain中前一个表进行连接的行数
Extra显示额外的信息

详细描述每一列的含义

id 列

id 列的值显示的是语句的执行顺序,id 列越大执行优先级越高,id 相同则从上往下执行,id 为 NULL 最后执行。

select_type 列

  1. simple:简单查询。查询不包含子查询和 union。
    EXPLAIN SELECT * FROM USER WHERE id=1;在这里插入图片描述
  2. primary:复杂查询中最外层的 select。
  3. subquery:包含在 select 中的子查询(不在 from 子句中)。
  4. derived:包含在 from 子句中的子查询。MySLQL 会将结果放在一个临时表中, 也称为派生表。
SET SESSION optimizer_switch='derived_merge=off'; 关闭对衍生表的合并
EXPLAIN SELECT (SELECT 1 FROM teacher WHERE id = 1) FROM (SELECT * FROM USER WHERE id = 1) der;

在这里插入图片描述

  1. union:在 union 中的第二个和以后的查询会被标为 union 类型。
  2. union result:从 union 构建的临时表检索结果的查询类型。
EXPLAIN SELECT id FROM teacher UNION SELECT id FROM teacher;

在这里插入图片描述

table 列

这列表示 explain 的一行正在访问哪个表,当 from 子句中有子查询时,table 列是 <derivenN> 格式,表示当前查询依赖 id=N 的查询,于是先执行 id=N 的查询。
当有 union 时,UNION RESULT 的 table 列的值为<union1,2>,1 和 2 表示参与 union 的 select 行 id。

partitions 列

如果查询的是基于分区的表,该字段显示查询将会访问的分区。

type 列

这一列表示关联类型或访问类型,即 MySQL 决定如何查找表中的行,查找数据记录的大概范围。
最优到最差:

system > const > eq_ref > ref > range > index > all

一般来讲,保证查询达到 range 级别,最好达到 ref
NULL:MySQL 能在优化阶段分解查询语句,在执行阶段不需访问表或索引。比如:

EXPLAIN SELECT MAX(id) FROM teacher

在这里插入图片描述
system:该表只有一行,是 const 的特例。
const:该表最多有一个匹配的行,MySQL 能对查询的某部分进行优化并将其转化成一个常量,因为只有一个匹配的行,所以速度非常快。

EXPLAIN SELECT * FROM (SELECT * FROM teacher WHERE id=1) tmp;

在这里插入图片描述
eq_ref:primary key 或 unique key 索引的所有部分被连接使用,最多只会返回一条符合条件的记录。

EXPLAIN SELECT * FROM teacher INNER JOIN teacher_user WHERE teacher.`id`=teacher_user.`teacher_id`;

在这里插入图片描述
ref:相比 eq_ref,不使用唯一索引,而是使用普通索引或者唯一性索引的部分前缀,索引要和某个值比较,可能会得到多个结果。

简单查询,name 是普通索引(非唯一索引)

EXPLAIN SELECT * FROM teacher WHERE NAME='t1';

在这里插入图片描述
关联表查询,idx_teacher_user_id 是 teacher_id 和 user_id 的联合索引,使用左边前缀 teacher_id 进行查询。

EXPLAIN SELECT teacher_id FROM teacher LEFT JOIN teacher_user ON teacher.id=teacher_user.teacher_id;

在这里插入图片描述
range:范围扫描通常出现在 in,between,>,<,>=等操作中,使用一个索引来检查给定范围的行。

EXPLAIN SELECT * FROM teacher WHERE id > 1;

在这里插入图片描述
index:扫描全索引就能拿到结果,一般是扫描某个二级索引,对二级索引的叶子节点进行遍历和扫描,所以速度较慢,而二级索引一般比较小,所以比 ALL 快。

EXPLAIN SELECT NAME FROM teacher;

在这里插入图片描述
ALL:全表扫描,扫描聚簇索引的所有叶子节点,通常这种情况需要增加索引进行优化。

EXPLAIN SELECT * FROM teacher;

在这里插入图片描述

possible_keys 列

这列显示的是此次查询可能用到的索引,一个或者多个,有时显示的是 NULL 值,是因为 MySQL 判断表中数据不多,不需要使用索引查询,选择全表查询。

如果该列是NULL,则没用相关索引。这种情况下,可以通过检查 where 子句看是否可以建立一个合适的索引来提高查询性能,再看 explain 的情况。覆盖索引查询的情况该列也为 NULL,但依然进行索引查询。

EXPLAIN SELECT NAME FROM teacher;

在这里插入图片描述

key 列

显示的是 MySQL 实际使用的索引。如果没有使用索引,则该列是 NULL,如果强制 MySQL 使用或忽视 possible_keys 列中的索引,在查询中使用 force index、ignore index。

EXPLAIN SELECT NAME FROM teacher IGNORE INDEX(idx_name);

在这里插入图片描述

key_len 列

这一列显示了 MySQL 在索引里使用的字节数,通过这个值可以算出具体使用了索引中的哪些列。它显示的是索引的最大长度,而不是实际使用长度。

EXPLAIN SELECT * FROM teacher_user WHERE teacher_id=1;

在这里插入图片描述
key_len 计算规则如下:

  • 字符串,char(n) 和 varchar(n),n 代表字符数,不是字节数,utf-8 的一个字母或数字占用 1 个字节,一个汉字占用 3 个字节
    • char(n):存汉字,长度为 3n 字节
    • varchar(n):存储汉字,长度为 3n + 2 字节,多的 2 字节用来存储字符串长度
  • 数值类型
    • tinyint:1 字节
    • smallint:2 字节
    • int:4 字节
    • bigint:8 字节
  • 时间类型
    • date:3 字节
    • timestamp:4 字节
    • datetime:8 字节
  • 如果字段允许为 NULL,需要 1 字节存储 NULL

ref 列

这一列显示了在 key 列记录的索引中,表查找值所用到的列或常量,常见的有 const(常量),字段名(例:teacher.id)

rows 列

这一列是 MySQOL 预估要读取并检测的行数,注意这个不是结果集里的行数

filtered 列

该列是一个百分比的值,根据 rows * filtered/100 可以估算出将要和 explain 中前一个表进行连接的行数。

Extra 列

常见的值如下:

  1. Using index:使用覆盖索引
EXPLAIN SELECT teacher_id FROM teacher_user WHERE teacher_id=1;

在这里插入图片描述
使用了联合索引 idx_teacher_user_id,覆盖索引查询索引覆盖的列,extra 列中显示 Using index。

  1. Using where:使用 where 语句来处理结果并且查询的列没被索引覆盖。
EXPLAIN SELECT * FROM teacher WHERE SUBJECT='语文';

在这里插入图片描述

  1. Using index condition:查询的列不完全被索引覆盖,where 条件中是一个前导列的范围。

  2. Using temporary:MySQL 需要创建一张临时表来处理查询。出现这种情况一般是要进行优化的,首先是想到用索引来优化。

EXPLAIN SELECT DISTINCT SUBJECT FROM teacher;

在这里插入图片描述

  1. Using filesort:将用外部排序而不是索引排序,数据较小时从内存排序,否则需要在磁盘完成排序。这种情况下一般也是需要考虑索引进行优化。
  2. Select tables optimized away:使用某些聚合函数来访问存在索引的某个字段。

总结

本文在结合查询例子的基础上对 Explain 工具查询的列进行讲解,很多内容都涉及到了,写到这也就差不多了,想要更详细的学习 Explain,可以去官网链接: MySQL 8 参考手册查看更详细的解释。

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

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

相关文章

第 1 章.提示词:开启AI智慧之门的钥匙

什么是提示词&#xff1f; 提示词&#xff0c;是引导语言模型的指令&#xff0c;让用户能够驾驭模型的输出&#xff0c;确保生成的文本符合需求。 ChatGPT&#xff0c;这位文字界的艺术大师&#xff0c;以transformer架构为基石&#xff0c;能轻松驾驭海量数据&#xff0c;编织…

R 生存分析3:Cox等比例风险回归及等比例风险检验

虽然Kaplan-Meier分析方法目前应用很广&#xff0c;但是该方法存在一下局限: 对于一些连续型变量&#xff0c;必须分类下可以进行生存率对比 是一种单变量分析&#xff0c;无法同时对多组变量进行分析 是一种非参数分析方法&#xff0c;必须有患者个体数据才能进行分析 英国…

阳光倒灌高准直汽车抬头显示器HUD太阳光模拟器

阳光倒灌高准直汽车抬头显示器HUD太阳光模拟器是一种高级别的模拟设备&#xff0c;用于模拟太阳光的光谱、强度及照射角度&#xff0c;应用于太阳能电池板、光伏系统等领域的研究和测试。其参数包括光谱范围、光强度、光源、照射角度、均匀性和稳定性&#xff0c;可根据需求调整…

ubuntu20.04安装截图工具flameshot

ubuntu20.04 自带的截图工具&#xff0c;可以使用快捷键“shift printScreen” ,但是它不能对截图进行编辑。 现在安装截图工具 flameshot&#xff0c;使用以下命令&#xff1a; sudo apt install flameshot 安装完成后&#xff0c;使用以下命令打开&#xff1a; flamesho…

Go 语言基础语法

目录 行分隔符 注释 标识符 字符串连接 关键字 Go 语言的空格 格式化字符串 Printf 实例 Go 语言变量 变量声明 多变量声明 值类型和引用类型 简短形式&#xff0c;使用 : 赋值操作符 Go 程序可以由多个标记组成&#xff0c;可以是关键字&#xff0c;标识符&#…

C++ 之LeetCode刷题记录(四十)

&#x1f604;&#x1f60a;&#x1f606;&#x1f603;&#x1f604;&#x1f60a;&#x1f606;&#x1f603; 开始cpp刷题之旅。 目标&#xff1a;执行用时击败90%以上使用 C 的用户。 27. 移除元素 给你一个数组 nums 和一个值 val&#xff0c;你需要 原地 移除所有数值…

文献速递:文献速递:基于SAM的医学图像分割--SAM-Med3D

Title 题目 SAM-Med3D 01 文献速递介绍 医学图像分析已成为现代医疗保健不可或缺的基石&#xff0c;辅助诊断、治疗计划和进一步的医学研究]。在这一领域中最重要的挑战之一是精确分割体积医学图像。尽管众多方法在一系列目标上展现了值得称赞的有效性&#xff0c;但现有的…

jmeter二次开发发送java请求_保姆级教程!!!

一、引言 JMeter是Apache基金会开发的一款开源性能测试工具&#xff0c;广泛应用于软件性能测试领域。它能够模拟多线程并发用户对应用程序进行压力测试&#xff0c;以评估应用程序的性能和稳定性。然而&#xff0c;在实际使用过程中&#xff0c;用户可能会遇到需要发送Java请…

RHCE实验-建立NFS服务器,使的客户端顺序共享数据

第一步&#xff1a;服务端及客户端的准备工作 # 恢复快照[rootserver ~]# setenforce 0​[rootserver ~]# systemctl stop firewalld​[rootserver ~]# yum install nfs-utils -y # 服务端及客户端都安装 第二步&#xff1a;服务端建立共享文件目录&#xff0c;并设置权限…

Vue3 新项目默认工程文件理解

Vue3 新项目默认工程文件理解 文章目录 Vue3 新项目默认工程文件理解0、工程文件结构图1、main.ts2、index.html源文件编译后 3、App.vue4、.d.ts 文件作用 0、工程文件结构图 1、main.ts // 引入 createApp 函数 import { createApp } from vue // 引入 style.css 文件&#…

【QT+QGIS跨平台编译】之九十五:【QGIS_App跨平台编译】—【错误处理:找不到标识符QwtPolarPoint】

文章目录 一、找不到标识符QwtPolarPoint二、原因分析三、错误处理一、找不到标识符QwtPolarPoint 报错信息如下: 二、原因分析 未找到QWT_POLAR_VERSION 从而执行的函数: mypMarker->setPosition( QwtPolarPoint( currentInfo.azimuth, currentInfo.elevation )

DevSecOps平台架构系列-微软云Azure DevSecOps平台架构

目录 一、概述 二、Azure DevOps和黄金管道 2.1 概述 2.2 Azure DevOps架构说明 2.2.1 架构及管道流程图 2.2.2 架构内容 2.2.2.1 Azure Boards 2.2.2.2 Azure Repos 2.2.2.3 Azure Test Plans 2.2.2.4 Azure Pipelines 2.2.2.5 Azure Application Insights 2.2.2.6…

浏览器https受信任证书生成——openssl颁发受信任证书

站点常常由于没有受信任的第三方CA机构颁发证书,使用https访问时,浏览器常常会弹出不安全的提示,为解决该问题,可以使用openssl颁发个人证书来解决该问题。 1openssl安装及使用方式参考:32.9 x509_OpenSSL 中文手册https://www.openssl.net.cn/docs/230.html2.本文章所有生…

java打家劫舍2(力扣Leetcode213)

打家劫舍2 力扣原题链接 问题描述 你是一个专业的小偷&#xff0c;计划偷窃沿街的房屋&#xff0c;每间房内都藏有一定的现金。这个地方所有的房屋都 围成一圈&#xff0c;这意味着第一个房屋和最后一个房屋是紧挨着的。同时&#xff0c;相邻的房屋装有相互连通的防盗系统&a…

相机显示储存卡未格式化怎么回事?怎么办

在摄影的学习和实践中&#xff0c;相机是我们记录美好瞬间的得力助手。然而&#xff0c;当相机突然提示储存卡未格式化时&#xff0c;这往往会让我们感到困惑和焦虑。本文将探讨相机显示储存卡未格式化的可能原因&#xff0c;并提供相应的解决方案。 图片来源于网络&#xff0c…

web自动化测试系列-selenium的运行原理和常用方法介绍(二)

目录 1.selenium的运行原理 2.常用方法介绍 接上文 &#xff1a;web自动化测试系列-selenium的安装和运行(一)-CSDN博客 在上文中我们编写了一段简单的代码 &#xff0c;可以驱动浏览器访问百度并搜索关键字 。这里我们再把这段代码再拿来加以说明 。 # 1. 导包 from selen…

搭建Spark单机版环境

在搭建Spark单机版环境的实战中&#xff0c;首先确保已经安装并配置好了JDK。然后&#xff0c;从群共享下载Spark安装包&#xff0c;并将其上传至目标主机的/opt目录。接着&#xff0c;解压Spark安装包至/usr/local目录&#xff0c;并配置Spark的环境变量&#xff0c;以确保系统…

数据分析 | pandas

主要数据结构&#xff1a; Series&#xff1a; Series 是一种类似于 Numpy 中一维数组的对象&#xff0c;它由一组任意类型的数据以及一组与之相关的数据标签&#xff08;即索引&#xff09;组成。 import pandas as pdprint(pd.Series([2, 4, 6, 8])) print(pd.Series([2, …

Python 全栈体系【四阶】(二十)

第五章 深度学习 二、推荐系统 1. 推荐算法介绍 1.1 个性化推荐算法 人口属性 地理属性 资产属性 兴趣属性 1.2 推荐算法分支 协同过滤推荐算法基于内容的推荐算法混合推荐算法流行度推荐算法 1.3 推荐算法 为推荐系统选择正确的推荐算法是非常重要的决定。目前为止…

基于springboot+vue调用百度ai实现车牌号识别功能

百度车牌号识别官方文档 结果视频演示 后端代码 private String getCarNumber(String imagePath, int count) {// 请求urlString url "https://aip.baidubce.com/rest/2.0/ocr/v1/license_plate";try {byte[] imgData FileUtil.readFileByBytes(imagePath);Stri…