mysql 执行计划

Posted 羽化登峰

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了mysql 执行计划相关的知识,希望对你有一定的参考价值。

下载mysql官网数据库模板:https://dev.mysql.com/doc/index-other.html
   

解压导入到数据库中  
    

官网查询执行计划地址: https://dev.mysql.com/doc/refman/5.5/en/explain-output.html
    

EXPLAIN输出列说明:
  id:为每个查询子句的标识符,为一组数字,表示查询中执行select子句或者操作表的顺序
    1)如果id相同,那么执行顺序从上到下

explain SELECT * FROM customer a join address b ON a.address_id = b.address_id;

    2)如果id不同,如果是子查询,id的序号会递增,id值越大优先级越高,越先被执行

explain SELECT * FROM customer a WHERE a.address_id = (SELECT b.address_id from address b WHERE b.address_id =100);

    3)id相同和不同的,同时存在:相同的可以认为是一组,从上往下顺序执行,在所有组中,id值越大,优先级越高,越先执行

explain SELECT * FROM customer a JOIN address c ON a.address_id = c.address_id WHERE a.address_id = (SELECT b.address_id from address b WHERE b.address_id =100);

  select_type:为每个查询子句类型
    1)SIMPLE 简单查询表

explain SELECT * FROM customer;

    2)primary:查询中若包含任何复杂的子查询,最外层查询则被标记为Primary,如以下语句a表

EXPLAIN SELECT a.* FROM customer a WHERE a.address_id = ( SELECT b.address_id FROM address b WHERE b.address_id = 100 ) ;

    3)union:若第二个select出现在union之后,则被标记为union,如以下语句b表

EXPLAIN SELECT a.first_name FROM customer a WHERE a.address_id = 100 UNION SELECT a.first_name FROM customer a WHERE a.address_id = 101;

    4)subquery:在select或者where列表中包含子查询,如以下语句b表

EXPLAIN SELECT a.* FROM customer a WHERE a.address_id = ( SELECT b.address_id FROM address b WHERE b.address_id = 100 ) ;

    5)MATERIALIZED:子查询物化,当表出现在非相关子查询中并且需要进行物化时会出现MATERIALIZED关键词

EXPLAIN SELECT a.* FROM customer a WHERE a.first_name IN ( SELECT DISTINCT b.first_name FROM customer b ) ;

    6)UNION RESULT:为union查询完成之后的结果集

EXPLAIN SELECT a.first_name FROM customer a WHERE a.address_id = 100 UNION SELECT a.first_name FROM customer a WHERE a.address_id = 101;

    7)DERIVED:from子句中出现的子查询,也叫做派生类

EXPLAIN SELECT c.* FROM (SELECT a.first_name FROM customer a WHERE a.address_id = 100 UNION SELECT b.first_name FROM customer b WHERE b.address_id = 101 ) c;

    8)DEPENDENT SUBQUERY:子查询b受a表影响

EXPLAIN SELECT (SELECT b.first_name FROM customer b WHERE b.first_name=a.first_name) FROM customer a;

    9)DEPENDENT UNION:子表b查询结果受a表影响,且在union后面

EXPLAIN SELECT (SELECT c.first_name FROM customer c WHERE c.customer_id =1 UNION SELECT b.first_name FROM customer b WHERE b.first_name=a.first_name) FROM customer a;

    10)UNCACHEABLE SUBQUERY:表示子查询不可被物化需要逐次运行

EXPLAIN select * from customer where customer_id = (select customer_id from customer where customer_id=@@sort_buffer_size);

    11)UNCACHEABLE UNION: 子查询中出现UNION并且不可被缓存在UNION 后的 SELECT 语句出现此关键词

EXPLAIN select * from customer where customer_id IN (select customer_id from customer WHERE customer_id=1 UNION select customer_id from customer where customer_id=@@sort_buffer_size);

    12)DEPENDENT DERIVED:衍生表受外部影响,未复现。
  table:为对应行正在访问哪一个表,表名或者别名,可能是临时表或者union合并结果集或衍生表
  partitions:分区表名
  type:查询数据方式,是全表查询还是索引查询检索
    效率从最好到最坏依次是:system > const > eq_ref > ref > ref_or_null > index_merge > unique_subquery > index_subquery > range > index > all
    1)all:全表扫描,一般情况下出现这样的sql语句而且数据量比较大的话那么就需要进行优化。

EXPLAIN SELECT * FROM customer;

    2)index:全索引扫描,这个比all效率要好,主要有两种情况可以使用:一种是当前sql是覆盖索引查询,第二种是进行索引排序

EXPLAIN SELECT customer_id FROM customer;

    3)range:利用索引进行范围查询。避免全表扫描,操作符: =, <>, >, >=, <, <=, IS NULL, BETWEEN, LIKE, or IN()

EXPLAIN SELECT customer_id FROM customer where customer_id>100 and customer_id <120;

    4)index_subquery:利用索引来关联子查询,不再扫描全表
    5)unique_subquery:该连接类型类似与index_subquery,使用的是唯一索引
    6)index_merge:在查询过程中需要多个索引组合使用,没有模拟出来
    7)ref_or_null:对于某个字段即需要关联条件,也需要null值的情况下
    8)fulltext:使用了全文索引
    9)ref:利用非唯一索引进行范围查询查询

EXPLAIN SELECT * FROM film WHERE title in (SELECT title FROM film WHERE title like \'a%\');

    10)eq_ref:使用唯一索引进行范围查询数据

EXPLAIN SELECT * FROM customer WHERE customer_id in (SELECT customer_id FROM customer);

    11)const:查询条件使用索引,且是常量值

EXPLAIN SELECT * FROM customer WHERE customer_id=100;

    12)system:表只有一行记录(等于系统表),这是const类型的特例,平时不会出现
  possible_keys : 可能使用的索引,当前表中所建立的索引
  key:实际使用的索引,如果未使用则显示为null
  key_len : 当前索引所使用的的字节数
  ref:显示索引的哪一列被使用了,或可能是一个常数
  rows:大致估算出找出所需记录需要读取的行数,此参数很重要,直接反应的sql找了多少数据,在完成目的的情况下越少越好
  filtered:查询数据与需要查询的结果比值
  extra:显示额外信息
    1)using filesort:说明未使用索引排序,在内存中使用排序算法进行排序
    2)using temporary:建立临时表来保存中间结果,查询完成后把临时表删除
    3)using index:表示当前的查询使用覆盖索引,
    4)using where:表示使用了索引,但是回表了
    5)using join buffer(hash join):表示查询结果在内存中创建hash临时表
    6)Impossible where:where语句的结果总是false

EXPLAIN SELECT a.* FROM film a WHERE 1 !=1;

以上是关于mysql 执行计划的主要内容,如果未能解决你的问题,请参考以下文章

MySQL系列- MySQL执行计划

Mysql学会查看sql的执行计划

mysql中如何查看优化器优化后的执行计划

带你看懂MySQL执行计划

mysql中如何查看优化器优化后的执行计划

mysql基础之三-;mysql执行计划