explain之type:访问类型
八种访问类型:
system、const、eq_ref、ref、range、index、all、null
最好到最差的顺序:
system>const>eq_ref>ref>range>index>all
最全的访问顺序:
system>const>eq_ref>ref>fulltext>ref_or_null>index_merge> unique_subquery>index_subquery>range>index>all
在项目使用中 至少优化到
range和ref
type之system:系统类型
表只有一行记录,等于系统表,这是常量const类型的特例
type之const:常量类型
常量,表示通过索引一次就找到了所查询的一条数据,
const用于此比较primary key(主键)或者unique(唯一)索引
因为只匹配一行数据所以查询很快
如将主键置于where列表中,mysql就能将该查询转换成一个常量
mysql> EXPLAIN SELECT * FROM( SELECT * FROM t1 WHERE id = 3 ) tmp1;
+----+-------------+------------+--------+---------------+---------+---------+-------+------+-------+
| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |
+----+-------------+------------+--------+---------------+---------+---------+-------+------+-------+
| 1 | PRIMARY | <derived2> | system | NULL | NULL | NULL | NULL | 1 | NULL |
| 2 | DERIVED | t1 | const | PRIMARY | PRIMARY | 4 | const | 1 | NULL |
+----+-------------+------------+--------+---------------+---------+---------+-------+------+-------+
t1 表访问类型是const常量,主键id在where列表中
最外层的select是临时表,只有一条数据所以是system类型
type之eq_ref:唯一或者主键索引查询
唯一索引扫描,对于每个索引建,
表中只有一条数据匹配,常见于主键或唯一索引扫描
使用索引查询,查询到的数据只有一条
mysql> explain select * from t1,t2 where t1.id = t2.id ;
+----+-------------+-------+--------+---------------+---------+---------+------------+------+-------+
| id | select_type | table | type | possible_keys | key | key_len | ref | rows | Extra |
+----+-------------+-------+--------+---------------+---------+---------+------------+------+-------+
| 1 | SIMPLE | t2 | ALL | PRIMARY | NULL | NULL | NULL | 1 | NULL |
| 1 | SIMPLE | t1 | eq_ref | PRIMARY | PRIMARY | 4 | demo.t2.id | 1 | NULL |
+----+-------------+-------+--------+---------------+---------+---------+------------+------+-------+
t1使用主键查询,只查询到一条数据,type是eq_ref
注意:t1与t2关联之后查出一条数据,type才会等于eq_ref
type之ref:where条件精确查询
非唯一索引扫描,返回匹配某个单独值得所有行
本质上也是一种索引访问,他返回所有匹配某个单独值得行
然而可能会找到多个符合条件的数据,所以他应该属于查找和扫描的混合体
匹配某个单独值得所有行,
组合索引,使用非全索引的字段去查询
查看索引:show index from t1;
创建索引(如果没有)create index inx_idcard_tel on t1 (id_card,tel)
mysql> explain SELECT * FROM t1 WHERE t1.id_card = '622421'\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: t1
type: ref
possible_keys: inx_idcard_tel
key: inx_idcard_tel
key_len: 57
ref: const
rows: 1
Extra: Using index condition
使用组合索引中的非全索引字段查询数据,type为ref
注意:这里不能单独使用tel字段查询,索引会失效
type之range:where范围查询
只检索指定范围的行,使用一个索引来进行选择行,key列显示使用了哪个字段的索引
一般就是在where语句中出现了between and 、>、<、in
等范围的查询条件
这种范围扫索引比全部索引扫描好一些,它只需要开始于索引的某个点,而结束语另一点,不用扫描全部索引
mysql> explain select * from t1 where t1.id between 1 and 10\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: t1
type: range
possible_keys: PRIMARY
key: PRIMARY
key_len: 4
ref: NULL
rows: 4
Extra: Using where
还有以下范围查询语句
explain SELECT * FROM t1 WHERE t1.id>1
explain SELECT * FROM t1 WHERE t1.id in(1,2);
type之index:全索引扫描
全索引扫描,是从索引中扫描全表
mysql> explain select id FROM t1\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: t1
type: index
possible_keys: NULL
key: inx_idcard_tel
key_len: 93
ref: NULL
rows: 4
Extra: Using index
这里是查询列中使用了主键索引,type=index
type之all:全表扫描
全表扫描,将遍历全表以找到匹配的行
mysql> explain select * FROM t1\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: t1
type: ALL
possible_keys: NULL
key: NULL
key_len: NULL
ref: NULL
rows: 4
Extra: NULL
防止使用*导致全表扫描
explain之possible_key与key:可能使用的索引和实际使用的索引
possible_key:
显示可能应用到这张表的索引,一个或者多个,查询涉及到的字段上若存在索引,则该索引被列出,但不一定被查询实际使用
key:
实际使用到的索引,如果为null,则没有使用索引,查询中如果使用了覆盖索引,则该索引仅出现在key列表中
覆盖索引:
查询列表中的字段要被所建的索引覆盖(小于等于)
情况一:理论上没有使用索引但实际上使用
mysql> explain select id FROM t1\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: t1
type: index
possible_keys: NULL
key: inx_idcard_tel
key_len: 93
ref: NULL
rows: 4
Extra: Using index
possible_keys=null
理论上没有使用索引key=inx_idcard_tel
实际上使用了组合索引注意:
这里使用到了覆盖索引的概念
情况二:理论上使用了索引但实际没有使用
mysql> explain select * from t1 where id_card=62 and tel='156'\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: t1
type: ALL
possible_keys: inx_idcard_tel
key: NULL
key_len: NULL
ref: NULL
rows: 4
Extra: Using where
possible_keys=inx_idcard_tel
理论上使用了索引key=null
实际上索引失效了,id_card是varchar类型,数值转字符串,导致索引失效注意:
varchar类型字段的数值必须使用单引号,防止索引失效
情况三:理论和实际上都没有使用索引
mysql> explain select * from t1 where tel='156'\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: t1
type: ALL
possible_keys: NULL
key: NULL
key_len: NULL
ref: NULL
rows: 4
Extra: Using where
possible_keys=null
理论上没有使用索引key=null
实际也没有使用注意:
组合索引的最左侧索引缺失,导致其他索引失效
情况四:理论和实际上都使用了索引
mysql> explain select * from t1 where id_card = '156' and tel='123'\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: t1
type: ref
possible_keys: inx_idcard_tel
key: inx_idcard_tel
key_len: 93
ref: const,const
rows: 1
Extra: Using index condition
mysql> explain select * from t1 where id_card = '156'\G
*************************** 1. row ***************************
id: 1
select_type: SIMPLE
table: t1
type: ref
possible_keys: inx_idcard_tel
key: inx_idcard_tel
key_len: 57
ref: const
rows: 1
Extra: Using index condition
理论和实际都使用了组合索引,
通过key_len可以判断出使用组合索引的单索引id_card
因为id_card和tel都使用了之后key_len=93
写在最后
人这一生也没有多少时间可以挥霍,
踏实一点,务实一些,
做自己想做的事,
如此简单!