前言
我們知道,索引的選擇是優(yōu)化器階段的工作,但是優(yōu)化器并不是萬能的,它有可能選錯(cuò)所要使用的索引。一般優(yōu)化器選擇索引考慮的因素有:掃描行數(shù),是否排序,是否使用臨時(shí)表。
使用explain分析sql
explain是很好的自測命令,勤于使用explain有助于我們寫出更合理的sql語句以及建立更合理的索引:
mysql> explain select * from t where (a between 1 and 1000) and (b between 50000 and 100000) order by b limit 1;
+----+-------------+-------+------------+-------+---------------+------+---------+------+-------+----------+------------------------------------+
| id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra |
+----+-------------+-------+------------+-------+---------------+------+---------+------+-------+----------+------------------------------------+
| 1 | SIMPLE | t | NULL | range | a,b | b | 5 | NULL | 50223 | 1.00 | Using index condition; Using where |
+----+-------------+-------+------------+-------+---------------+------+---------+------+-------+----------+------------------------------------+
1 row in set, 1 warning (0.01 sec)
其中:
table字段:表示關(guān)于哪張表;
type字段:system,const,eq_reg,ref,range,index,all。一般來說要達(dá)到range級(jí)別以上;
system、const:可以將查詢的變量轉(zhuǎn)為常量,如id=1;id為主鍵或唯一鍵;
eq_ref:訪問索引,返回某單一行的數(shù)據(jù),通常在連接時(shí)出現(xiàn),查詢使用的索引為主鍵或唯一鍵;
ref:訪問索引,返回某個(gè)值得數(shù)據(jù)(可能是多行),通常使用=時(shí)發(fā)生;
range:使用索引返回一個(gè)范圍內(nèi)的行信息,如使用>,,between
index:以索引的順序進(jìn)行全表掃描,雖然有索引不用排序,但是要全表掃描;
all:全表掃描
key字段:實(shí)際使用的索引;
key_len字段:使用的索引長度(在不損失精度的情況下,長度越短越好);
ref字段:顯示索引的哪一列被使用了;
rows字段:MySQL認(rèn)為檢索需要的數(shù)據(jù)行數(shù);
Extra字段:查詢的額外信息,主要有以下幾種:
using index:使用了索引
using where:使用了where條件
using tmporary:用到臨時(shí)表去處理當(dāng)前查詢
using filesort:用到額外的排序,如order字段無索引
range checked for eache record(index map:N):無索引可用
using index for group-by:表名可以在索引中找到分組所需的所有數(shù)據(jù),不需要查詢實(shí)際的表
一般遇到Using temporary和Using filesort就要想辦法優(yōu)化一下了,因?yàn)橛貌坏剿饕?/p>
MySQL怎么計(jì)算需要檢索的行數(shù)
實(shí)際中,MySQL所統(tǒng)計(jì)的掃描行數(shù)并不是精確值,有時(shí)候甚至?xí)嗖詈苓h(yuǎn),而掃描行數(shù)則是基于索引的基數(shù)來計(jì)算的。
在MySQL中,通過采樣統(tǒng)計(jì)的方式去獲取索引基數(shù):系統(tǒng)默認(rèn)選取 N 個(gè)數(shù)據(jù)頁,統(tǒng)計(jì)數(shù)據(jù)頁上不同值得平均值,然后乘以索引的頁面數(shù)得到基數(shù),而且MySQL會(huì)在變更的數(shù)據(jù)行數(shù)超過 1/M 時(shí)來觸發(fā)重做索引統(tǒng)計(jì)的操作。
在MySQL中,有2種存儲(chǔ)索引統(tǒng)計(jì)的方式,可以通過設(shè)置innodb_stats_persistent參數(shù)來選擇:
設(shè)置為 on 的時(shí)候,表示統(tǒng)計(jì)信息會(huì)持久化存儲(chǔ)。這時(shí),默認(rèn)的 N 是 20,M 是 10。
設(shè)置為 off 的時(shí)候,表示統(tǒng)計(jì)信息只存儲(chǔ)在內(nèi)存中。這時(shí),默認(rèn)的 N 是 8,M 是 16。
一般來說,基數(shù)統(tǒng)計(jì)出來的數(shù)據(jù)和真實(shí)的行數(shù)沒有很大差距,但是涉及到刪除數(shù)據(jù)新增數(shù)據(jù)比較頻繁的數(shù)據(jù)表,可能會(huì)出現(xiàn)數(shù)據(jù)表有10萬條數(shù)據(jù)但是基數(shù)統(tǒng)計(jì)卻有20萬的情況,這就可能是MVCC在作怪了,因?yàn)镸ySQL的InnoDB的事務(wù)支持,需要維持多個(gè)數(shù)據(jù)版本,就有可能某些事務(wù)還沒結(jié)束,還在使用刪除了很久的數(shù)據(jù)導(dǎo)致已刪除的數(shù)據(jù)空間無法釋放,而新增的數(shù)據(jù)又開辟了新的空間,那么這時(shí)候就導(dǎo)致基數(shù)統(tǒng)計(jì)中數(shù)據(jù)頁數(shù)量可能出現(xiàn)失誤,出現(xiàn)較大誤差。
一個(gè)很好的修正方式就是執(zhí)行analyze table 表名,該命令用來重新統(tǒng)計(jì)索引信息。
索引選錯(cuò)了我們到底怎么辦
當(dāng)我們正確的建立必須的索引后,大部分情況下,優(yōu)化器其實(shí)并不會(huì)選擇錯(cuò)索引,當(dāng)我們遇到索引選錯(cuò)的情況下,該怎么去處理呢?
1、使用force index強(qiáng)制使用某個(gè)索引。
2、轉(zhuǎn)換思路,優(yōu)化一下sql語句可能就會(huì)使用到該使用的索引。
3、新建更合適的索引或刪除掉誤用到的不合理的索引。(有些時(shí)候,可能真的是這個(gè)索引是多余的,還不是最優(yōu)的,優(yōu)化器又剛好使用到了它)。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。
您可能感興趣的文章:- mysql 添加索引 mysql 如何創(chuàng)建索引
- MySQL索引類型總結(jié)和使用技巧以及注意事項(xiàng)
- MySQL查看、創(chuàng)建和刪除索引的方法
- MySQL 創(chuàng)建索引(Create Index)的方法和語法結(jié)構(gòu)及例子
- MySQL 主鍵與索引的聯(lián)系與區(qū)別分析
- Mysql索引會(huì)失效的幾種情況分析
- 基于mysql全文索引的深入理解
- 解決MySQL中IN子查詢會(huì)導(dǎo)致無法使用索引問題
- Mysql中的Btree與Hash索引比較
- MYSQL中常用的強(qiáng)制性操作(例如強(qiáng)制索引)