慢sql查询优化

explain使用介绍

id:执行编号,标识select所属的行。如果在语句中没子查询或关联查询,只有唯一的select,每行都将显示1。否则,内层的select语句一般会顺序编号,对应于其在原始语句中的位置

select_type:显示本行是简单或复杂select。如果查询有任何复杂的子查询,则最外层标记为PRIMARY(DERIVED、UNION、UNION RESUlT)

table:访问引用哪个表(引用某个查询,如“derived3”)

type:数据访问/读取操作类型(ALL、index、range、ref、eq_ref、const/system、NULL)

possible_keys:揭示哪一些索引可能有利于高效的查找

key:显示mysql决定采用哪个索引来优化查询

key_len:显示mysql在索引里使用的字节数

ref:显示了之前的表在key列记录的索引中查找值所用的列或常量

rows:为了找到所需的行而需要读取的行数,估算值,不精确。通过把所有rows列值相乘,可粗略估算整个查询会检查的行数。整编:微信公众号,搜云库技术团队,ID:souyunku

Extra:额外信息,如using index、filesort等

            Using  filesort表示通过对返回数据进行排序

扫描二维码关注公众号,回复: 8252515 查看本文章

            Using temporary表示查询有使用临时表, 一般出现于排序, 分组和多表join的情况, 查询效率不高, 仍需要进行优化,出现临时表的原因还可能是数据量过大使用了临时表进行分组运算

重点关注type,type类型的不同竟然导致性能差六倍!!!

type显示的是访问类型,是较为重要的一个指标,结果值从好到坏依次是:system > const > eqref > ref > fulltext > refornull > indexmerge > uniquesubquery > indexsubquery > range > index > ALL ,一般来说,得保证查询至少达到range级别,最好能达到ref。

All:最坏的情况,全表扫描

index:和全表扫描一样。只是扫描表的时候按照索引次序进行而不是行。主要优点就是避免了排序, 但是开销仍然非常大。如在Extra列看到Using index,说明正在使用覆盖索引,只扫描索引的数据,它比按索引次序全表扫描的开销要小很多

range:范围扫描,一个有限制的索引扫描。key 列显示使用了哪个索引。当使用=、 <>、>、>=、<、<=、IS NULL、<=>、BETWEEN 或者 IN 操作符,用常量比较关键字列时,可以使用 range |

ref:一种索引访问,它返回所有匹配某个单个值的行。此类索引访问只有当使用非唯一性索引或唯一性索引非唯一性前缀时才会发生。这个类型跟eq_ref不同的是,它用在关联操作只使用了索引的最左前缀,或者索引不是UNIQUE和PRIMARY KEY。ref可以用于使用=或<=>操作符的带索引的列。

eq_ref:最多只返回一条符合条件的记录。使用唯一性索引或主键查找时会发生 (高效)

const:当确定最多只会有一行匹配的时候,MySQL优化器会在查询前读取它而且只读取一次,因此非常快。当主键放入where子句时,mysql把这个查询转为一个常量(高效)

system:这是const连接类型的一种特例,表仅有一行满足条件。

Null:意味说mysql能在优化阶段分解查询语句,在执行阶段甚至用不到访问表或索引(高效)

举例SQL: EXPLAIN select * from student  group by    age 

解决方案:

1、尽量避免在where子句中对字段进行函数操作,这将导致存储引擎放弃使用索引而进行全表扫描。对于需要计算的值最好通过程序计算好传入而不是在sql语句中做计算

2、group by实质是先排序后分组,也就是分组之前必排序。通过分组的时候禁止排序优化sql 

举例:group by name order by null

3、使用分组查询、范围查询,而不是全量查询

猜你喜欢

转载自www.cnblogs.com/wueryuan/p/12072248.html