`

MySQL Explain 详解

阅读更多

 

该贴转自:MySQL Explain 详解

id 列

从ID列可以判断SQL的执行顺序:从大到小,如果ID相同,就从上到下执行。
如果行引用联合结果的其他行,那么它的值可能为NULL,这种情况下,table列的值会显示为<unionM,N>来指明行引用的是联合行中的带有指定M,N值的ID。

select_type 列

SIMPLE

简单的SELECT语句(没有使用 union 或者 子查询)

PRIMARY

最外层的SELECT语句

UNION

在一个UNION语句中第二或之后的SELECT语句

DEPENENT_UNION

在一个UNION语句中第二或之后的SELECT语句,取决于外层的查询

UNION_RESULT

UNION的结果集

SUBQUERY

子查询中的第一个SELECT

DEPENENT_SUBQUERY

子查询中的第一个SELECT,取决于外层的查询

DERIVED

SELECT的驱动表(FROM子查询)

MATERIALIZED

物化子查询

UNCACHEABLE SUBQUERY

不能将结果缓存的子查询,必须重新计算外部查询的每一行

UNCACHEABLE UNION

在一个UNION中第二或之后的SELECT查询属于UNCACHEABLE SUBQUERY(请看UNCACHEABLE SUBQUERY)

特别说明

DEPENDENT:典型代表就是使用了相关子查询。(相关子查询:子查询里包含了一个同时在子查询里,又在外部查询的表的查询)

DEPENDENT SUBQUERY: 它不同于UNCACHEABLE SUBQUERY的求值。对于DEPENDENT SUBQUERY,子查询对于外部上下文里每一个集合中不同的变星值仅仅重新计算一次。

而对于UNCACHEABLE SUBQUERY,子查询对于外部上下文里的每一行都会重新计算一次.

table 列

SQL中使用到的表.它的值也可为下面的其中之一:

<unionM,N>: 行引用了带有值为M,N的查询ID的联合行

<derivedN>: 行引用了值为N的ID的驱动表结果。例如,它是从FROM子查询中的结果形成的驱动表的结果集

<subqueryN>: 行引用了一个值为N的ID的materialized subquery(物化子查询)的结果。

partitions 列

查询中所匹配的记录所在的分区。这列只有在使用了PARTITIONS关键字时才会显示。对于没有使用分区表的,该值为NULL

type 列

连接类型

system

表只有一行(=system table).这个const连接类型的一个特例。

const

该表最多只有一条匹配的行,这是读取查询的开始。因为这里只有一行,该行这列的值可以被优化器的剩余部分认为是常量。const表是非常快的,因为它们只会读取一次。

const用于当你与主键或唯一索引的所有部分比较的是常量值时。在以下的查询中,tal_name 可以被用作const表:

1
2
3
select * from tbl_name where primary_key = 1;

select * from tbl_name where primary_key_part1 = 1 and primary_key_part2 = 2

eq_ref

从该表中读取一行与前一张表的所有行的每一种组合。除了systemconst类型,这可能是最好的连接类型了。它通常于用在连接时使用了索引的所有部分,并且索引是一个主键索引或者唯一非空索引.

eq_ref可用于索引列使用=比较操作符的情况。比较的值可能是一个常量或者是 从读这表之前的表中使用的一个列表达式。比如下面的例子,MySQL能够使用eq_ref连接去处理ref_table表:

1
2
3
select * from ref_table, other_table where ref_table.key_column 
= other_table.column;

select * from ref_table, other_table where ref_table.key_column_part1
 = other_table.column and ref_table.key_column_part2 = 1

ref

所有索引匹配的行的值与前一张表的所有行的每一种组合。ref用于连接仅使用最左前缀索引或者索引不是主键索引,唯一索引(换句话说,连接不能基于该索引值选择一行 )。如果索引用于匹配少数行,这是一个好的连接类型。

ref能够用于使用=或者<=>操作符的索引列中。比如下面的例子,MySQL能够使用ref连接去处理ref_table表:

1
2
3
4
5
select * from ref_table where key_column = expr;

select * from ref_table , other_table where ref_table.key_column
 = other_table.column;

select * from ref_table, other_table where ref_table.key_column_part1
 = other_table.column and ref_table.key_column_part2 = 1;

fulltext

这个连接使用fulltext索引执行

ref_or_null

这个连接类型是类似ref,但有些额外的不同:MySQL要做一些额外操作去搜索包含NULL值的行。这个连接类型通常用于优化执行子查询。在以下的例子里,MySQL 可以使用ref_or_null连接去处理ref_table.

1
select * from ref_table where key_column = expr or key_column is null;

index_merge

这个连接类型说明使用了索引合并优化。在这种情况下,在输出行的key列包含了一个使用索引的列表,并且key_len包含了使用了索引最长的那部分列表。

unique_subquery

这个类型在以下格式的一些IN子查询是替代ref类型的。

value IN (select primary_key from single_table where some_expr)

unique_subquery 完全只是为了更高效地替代子查询的一个索引查找函数。

index_subquery

这个连接类型是类似unique_subquery。它代替IN子查询,但用于以下格式的子查询中的非唯一索引

value in (select key_column from single_table where some_expr)

range

仅在给定的范围使用索引检索行。在输出的行中的key列说明使用了哪个索引。key_len列包含了使用了索引的最长部分。对于这类型下,ref列的值为NULL

range可以用于当索引列使用以下任一个操作符与常量比较时使用:=<>>>=<<=IS NULL<=>,BETWEEN 或者 IN()

1
2
3
4
5
6
7
select * from tbl_name where key_column = 10;

select * from tbl_name where key_colum between 10 and 20;

select * from tbl_name where key_column in (10, 20, 30);

select * from tbl_name where key_part1 = 10 and key_part2 in (10,20,30);

index

index索引连接类型与all相同,除了它是扫描索引树外(即全索引扫描)。这在两种情况下发生:

  1. 如果索引对于查询来说是覆盖索引并且可用于满足要求表的数据要求,这时只有索引树会被扫描。在这种情况下,Extra列会显示为Using index。一个只读索引扫描通常比ALL更快,因为索引的大小通常是比表的大小更小。

  2. 全表扫描的执行是从索引读取的顺序去查找数据行。这时Extra列不会出现Uses index

MySQL 当查询仅使用单独索引的一部分列时才会使用这个连接类型。

ALL

从前表中每一个行组合进行全表扫描。如果表是第一个表并没有标识为const的话,这通常是不好的,并且在其他的情况下是非常糟糕的。通过,你可以通过添加索引来避免ALL连接类型,这使得行检索基于常量值或者从更早的表中的列值。

possible_keys 列

该列指出MySQL可以从该表中使用哪个索引去查找行。注意,该列完全独立于explain输出结果显示的表的顺序的。这意味着,一些在possible_keys里的键,在实际上可能无法在产生表的顺序时使用。

如果没有相关的索引,该列就为NULL。在这种情况下,你可以通过测试WHERE子句检查它使用的列有没有适当的索引来提高查询的性能。如果的确如此的话,创建一个合适的索引并使用Explain来再次检查你的查询有没有用上了索引。

查看表有哪些索引,可以使用show index from tbl_name

key 列

该列指出MySQL实际决定使用的key(索引)。如果MySQL决定使用possible_keys中之一的索引去查找行,那么那个索引的值就是该key列的值

不过,有可能key列的索引名没有出现在possible_keys列中。这可能发生在,如果possible_keys索引中没有一个是适合查找行的索引,但查询所选择的列都是其他索引中包含的列的情况。这意味着,命名的索引覆盖了所选择的列,所以尽管它不用于决定哪些行要检索,但一个索引扫描比数据行扫描更高效。

对于InnoDB,二级索引可能覆盖了所选择的列,尽管查询也选择了主键,因为InnoDB保存每一个二级索引里保存了主键的值。如果key列为NULL,MySQL会发现没有索引可用于执行更高效的查询。

强制MySQL去使用或者忽略在possible_keys列的某个索引,可以在查询中使用FORCE INDEX, USE INDEX 或者 IGNORE INDEX

对于MyISAMNDB表,执行ANALYZE TABLE可以帮助优化器选择更好的索引。对于NDB表,这也会提高分布式下推连接的性能。对于MyISAM表,myisamchk --analyze 与 analyze table的作用相同。

key_len 列

该列指出MySQL决定使用的索引(key)的长度。如果key列为NULL,那么该列也为NULL。注意,key_len的值能够让你确定MySQL实际上使用了一个组合索引中的多少部分。

ref 列

该列显示了哪些列或常量是用于与在key列的命名的索引列来进行比较来从表中选择行的。

如果值为func,该值就会用于一些函数的结果集。为了查找出哪些函数,执行SHOW WARNINGS之后,使用EXPLAIN EXTENDED来查看。函数可能实际上是一个操作符,像算术操作那样。

rows 列

该列指出MySQL认为它必须检验执行查询的行数。

对于InnoDB表,这个数值是个估计值,并不总是准确的。

filtered 列

该列指出,表行中的百分比将会被表的条件过滤。即,rows列显示了测试的行数,而(rows * filtered / 100) 显示了将与之前的表进行连接的行数。该列将会在你使用explain extended`时显示出来。

extra 列 及常见的值

该列包含了额外的关于MySQL执行查询的信息。

const row not found

对于像select ... from tbl_name这样的查询,该表是空的。

Deleting all rows

对于DELETE,一些存储引擎(例如MyISAM)支持的一个处理方法简单并且快速地删除所有表行。如果引擎使用了这个优化,则Extra表的值就会显示为这个。

Distinct

MySQL 正查找不同复的值,所以,它会停止搜索当前行之后已经第一次匹配的行。

FirstMatch(tbl_name)

它是用于表tbl_name的 semi-join FirstMatch策略的缩写

Full scan on NULL key

这出现在当优化器不能使用索引来优化子查询时的一个备用策略。

Impossible HAVING

HAVING子句总是为false并且不能检索到任何行。

Impossible WHERE

WHERE子句总是为false并且不能检索到任何行。

Impossible WHERE noticed after reading const tables

MySQL 已经读取了所有const(和system)表,并且注意到WHERE子句总是为false.

LooseScan(m…n)

semi-join LooseScan策略被使用。m和n是索引部分的数字。

Materialize, Scan

在MySQL 5.6.7 之前,这表明使用了一个独立的物化临时表。如果Scan出现,表示临时表索引没有用于读取表。否则,表示使用了索引来查找。同样也看Start materialize

在 MySQL 5.6.7 中,物化说明行的select_type列值为MATERIALIZED,并且行的table表是一个值为<subqueryN>

No matching min/max row

对于类似select min(...) from ... where condition这样的查询没有满足条件的行。

no matching row in const table

对于一个连接查询,有一个空表或者一个没有满足唯一索引条件的行的表。

No matching rows after partition pruning

对于DELETE或者UPDATE,优化器发现进行分区调整时发现没有数据可以删除或者更新。这类似于select语句的Impossible WHERE

No tables used

查询没有from子句,或者有一个from dual子句。

对于INSERT或者REPLACE语句,当没有select部分时explain就会显示这个值。例如, 它会出现在explain insert into t values(10),因为它等效于explain insert into t select 10 from dual.

Not exists

MySQL能够在进行left join查询优化,在它发现匹配LEFT JOIN 标准一行数之后不会对于之前的行组合中在该表上检查更多的行。这有一个使用该优化的查询例子:

select * from t1 left join t2 on t1.id = t2.id where t2.id is null

假设t2.id是定义为NOT NULL的。在这种情况下,MySQL扫描t1并使用t1.id的值查找t2的行。如果MySQL发现在t2中有匹配的行,它知道t2.id能够永不为NULL,并且不会扫描t2中具有相同id值的剩余的行。换句话说,对于每一个t1的行,MySQL只需要在t2中进行一次查找,而不管在t2中实际有多少匹配的行。

Using filesort

MySQL 必须做一些额外的工作去决定如何检索的行进行排序。排序是通过根据连接类型所筛选的所有行然后保存要排序的键和这些所有经过WHERE子句筛选的所有行的指针来完成的。

这些排序键将被保存然后按排序的顺序来检索行数据。

Using index

从表中检索的列信息只需从索引信息中获取,而不用经过额外的访问实际的表数据来寻找。这个策略

分享到:
评论

相关推荐

    MySQL explain详解.md

    MySQL explain详解

    Mysql Explain 详解.txt

    标题:Mysql Explain 详解 描述:深入解析MySQL的Explain功能,理解其在查询优化中的关键作用。 知识点详述: ### 一、Explain 的基本概念与使用 #### 1.1 概述 `Explain`是MySQL提供的一种用于分析SQL执行计划...

    MYSQL EXPLAIN详解

    ### MySQL EXPLAIN详解 #### 一、EXPLAIN简介 **EXPLAIN** 是 MySQL 提供的一个非常有用的工具,它能够帮助用户了解 MySQL 如何执行查询语句,并展示出 MySQL 预估的查询计划。这对于理解 SQL 查询的工作原理、...

    mysql explain详解

    MySQL中的`EXPLAIN`命令是用于分析SQL查询执行计划的重要工具,它可以帮助数据库管理员和开发者了解MySQL如何处理SQL语句,从而进行性能优化。在深入理解`EXPLAIN`之前,我们先来了解一下数据库优化的基本概念。 ...

    mysql explain详解1

    MySQL的EXPLAIN命令是数据库管理员和开发者用来分析SQL查询执行计划的重要工具,它能帮助我们理解MySQL如何使用索引来处理SELECT语句,并优化查询性能。本文将深入解析EXPLAIN输出的各项参数,以便更好地理解和优化...

    Mysql Explain 详解.docx

    MySQL中的`EXPLAIN`命令是用于分析SQL查询执行计划的重要工具,它可以帮助数据库管理员和开发者了解MySQL如何处理SQL语句,从而优化查询性能。在处理慢查询时,`EXPLAIN`是不可或缺的辅助手段。 一、`EXPLAIN`基本...

    02-VIP-Explain详解与索引最佳实践.pdf

    Mysql Explain 详解与索引最佳实践 Mysql Explain 是 Mysql 中一个非常重要的工具,可以帮助我们分析和优化 SQL 查询语句。在本文中,我们将详细介绍 Mysql Explain 的使用和解释 Explain 输出结果,帮助您更好地...

    MySQL-Explain 详解

    MySQL Explain 详解 MySQL Explain 是 MySQL 中的一个重要工具,用于解释和优化 SQL 查询语句。通过 Explain,可以了解 MySQL 是如何执行查询语句的,并且可以根据 Explain 的结果来优化查询语句。 一、用法 ...

    Mysql Explain详细解析

    ### MySQL EXPLAIN 详解 #### 一、EXPLAIN 概述 MySQL 的 `EXPLAIN` 命令是一个非常强大的工具,它可以帮助我们理解 MySQL 如何执行查询,并为我们提供优化查询性能的重要信息。通过 `EXPLAIN`,我们可以了解查询...

    mysql 中explain的详解

    EXPLAIN关键字一般放在SELECT查询语句的前面,用于描述MySQL如何执行查询操作、以及MySQL成功返回结果集需要执行的行数。explain 可以帮助我们分析 select 语句,让我们知道查询效率低下的原因,从而改进我们查询,让...

    Explain详解及SQL优化实战.docx

    ### MySQL Explain 详解及SQL优化实战 #### 一、Explain 概述 在进行数据库查询优化时,了解SQL查询的执行计划是非常重要的一步。`EXPLAIN` 是MySQL提供的一个非常有用的工具,它可以帮助我们理解MySQL如何执行一...

    Mysql Explain

    #### 三、EXPLAIN 输出详解 `EXPLAIN` 的输出结果提供了关于查询执行计划的详细信息。每一个输出行都包含了一个表的相关信息,以及一系列的列,这些列用于解释查询的具体行为。 1. **id** - 表示查询中执行 `...

    Explain详解与索引最佳实践.zip

    "Explain详解与索引最佳实践"的主题主要聚焦于如何利用MySQL的`EXPLAIN`命令来理解并优化SQL查询,以及如何有效地设计和利用索引来提升查询效率。 `EXPLAIN`是MySQL提供的一种强大工具,用于分析查询执行计划。当你...

    Explain详解与索引最佳实践.docx

    在MySQL数据库管理中,了解如何使用`EXPLAIN`关键字对于优化SQL查询性能至关重要。`EXPLAIN`可以帮助我们分析查询执行计划,揭示MySQL是如何处理SQL语句的,从而找到潜在的性能瓶颈。本文档主要探讨了`EXPLAIN`的...

    MySQL explain 笔记整理

    #### 二、EXPLAIN 输出字段详解 在使用 `EXPLAIN` 分析 SQL 执行计划时,会得到一系列的输出字段,下面将详细介绍这些字段的意义: - **id**:表示 SELECT 查询的顺序号。如果一个查询包含多个部分(如子查询),...

    Explain详解与索引实践.pdf

    使用EXPLAIN关键字可以模拟优化器执行SQL语句,分析你的查询语句或是结构的性能瓶颈 在 select 语句之前增加 explain 关键字,MySQL 会在查询上设置一个标记,执行查询会返 回执行计划的信息,而不是执行这条...

    Explain详解与索引最佳实践.pdf

    ### EXPLAIN详解与索引最佳实践 #### 一、Explain 概述 在数据库管理与优化领域,理解和掌握SQL查询执行的过程对于提高查询效率至关重要。`EXPLAIN` 是 MySQL 提供的一种工具,用于帮助数据库管理员及开发人员了解...

Global site tag (gtag.js) - Google Analytics