概述MySQL統(tǒng)計(jì)信息
MySQL執(zhí)行SQL會(huì)經(jīng)過(guò)SQL解析和查詢(xún)優(yōu)化的過(guò)程,解析器將SQL分解成數(shù)據(jù)結(jié)構(gòu)并傳遞到后續(xù)步驟,查詢(xún)優(yōu)化器發(fā)現(xiàn)執(zhí)行SQL查詢(xún)的最佳方案、生成執(zhí)行計(jì)劃。查詢(xún)優(yōu)化器決定SQL如何執(zhí)行,依賴(lài)于數(shù)據(jù)庫(kù)的統(tǒng)計(jì)信息,下面我們介紹MySQL 5.7中innodb統(tǒng)計(jì)信息的相關(guān)內(nèi)容。
MySQL統(tǒng)計(jì)信息的存儲(chǔ)分為兩種,非持久化和持久化統(tǒng)計(jì)信息。
一、非持久化統(tǒng)計(jì)信息
非持久化統(tǒng)計(jì)信息存儲(chǔ)在內(nèi)存里,如果數(shù)據(jù)庫(kù)重啟,統(tǒng)計(jì)信息將丟失。有兩種方式可以設(shè)置為非持久化統(tǒng)計(jì)信息:
1 全局變量, INNODB_STATS_PERSISTENT=OFF |
2 CREATE/ALTER表的參數(shù), STATS_PERSISTENT=0 |
非持久化統(tǒng)計(jì)信息在以下情況會(huì)被自動(dòng)更新:
1 執(zhí)行ANALYZE TABLE |
2 innodb_stats_on_metadata=ON情況下,執(zhí)SHOW TABLE STATUS, SHOW INDEX, 查詢(xún) INFORMATION_SCHEMA下的TABLES, STATISTICS |
3 啟用--auto-rehash功能情況下,使用mysql client登錄 |
4 表第一次被打開(kāi) |
5 距上一次更新統(tǒng)計(jì)信息,表1/16的數(shù)據(jù)被修改 |
非持久化統(tǒng)計(jì)信息的缺點(diǎn)顯而易見(jiàn),數(shù)據(jù)庫(kù)重啟后如果大量表開(kāi)始更新統(tǒng)計(jì)信息,會(huì)對(duì)實(shí)例造成很大影響,所以目前都會(huì)使用持久化統(tǒng)計(jì)信息。
二、持久化統(tǒng)計(jì)信息
5.6.6開(kāi)始,MySQL默認(rèn)使用了持久化統(tǒng)計(jì)信息,即INNODB_STATS_PERSISTENT=ON,持久化統(tǒng)計(jì)信息保存在表mysql.innodb_table_stats和mysql.innodb_index_stats。
持久化統(tǒng)計(jì)信息在以下情況會(huì)被自動(dòng)更新:
1 INNODB_STATS_AUTO_RECALC=ON 情況下,表中10%的數(shù)據(jù)被修改 |
2 增加新的索引 |
innodb_table_stats是表的統(tǒng)計(jì)信息,innodb_index_stats是索引的統(tǒng)計(jì)信息,各字段含義如下:
innodb_table_stats |
|
database_name |
數(shù)據(jù)庫(kù)名 |
table_name |
表名 |
last_update |
統(tǒng)計(jì)信息最后一次更新時(shí)間 |
n_rows |
表的行數(shù) |
clustered_index_size |
聚集索引的頁(yè)的數(shù)量 |
sum_of_other_index_sizes |
其他索引的頁(yè)的數(shù)量 |
innodb_index_stats |
|
database_name |
數(shù)據(jù)庫(kù)名 |
table_name |
表名 |
index_name |
索引名 |
last_update |
統(tǒng)計(jì)信息最后一次更新時(shí)間 |
stat_name |
統(tǒng)計(jì)信息名 |
stat_value |
統(tǒng)計(jì)信息的值 |
sample_size |
采樣大小 |
stat_description |
類(lèi)型說(shuō)明 |
為更好的理解innodb_index_stats,建一張測(cè)試表做說(shuō)明:
CREATE TABLE t1 ( a INT, b INT, c INT, d INT, e INT, f INT, PRIMARY KEY (a, b), KEY i1 (c, d), UNIQUE KEY i2uniq (e, f) ) ENGINE=INNODB;
寫(xiě)入數(shù)據(jù)如下:
查看t1表的統(tǒng)計(jì)信息,需主要關(guān)注stat_name和stat_value字段
tat_name=size時(shí):stat_value表示索引的頁(yè)的數(shù)量
stat_name=n_leaf_pages時(shí):stat_value表示葉子節(jié)點(diǎn)的數(shù)量
stat_name=n_diff_pfxNN時(shí):stat_value表示索引字段上唯一值的數(shù)量,此處做一下具體說(shuō)明:
1、n_diff_pfx01表示索引第一列distinct之后的數(shù)量,如PRIMARY的a列,只有一個(gè)值1,所以index_name='PRIMARY' and stat_name='n_diff_pfx01'時(shí),stat_value=1。
2、n_diff_pfx02表示索引前兩列distinct之后的數(shù)量,如i2uniq的e,f列,有4個(gè)值,所以index_name='i2uniq' and stat_name='n_diff_pfx02'時(shí),stat_value=4。
3、對(duì)于非唯一索引,會(huì)在原有列之后加上主鍵索引,如index_name='i1' and stat_name='n_diff_pfx03',在原索引列c,d后加了主鍵列a,(c,d,a)的distinct結(jié)果為2。
了解了stat_name和stat_value的具體含義,就可以協(xié)助我們排查SQL執(zhí)行時(shí)為什么沒(méi)有使用合適的索引,例如某個(gè)索引n_diff_pfxNN的stat_value遠(yuǎn)小于實(shí)際值,查詢(xún)優(yōu)化器認(rèn)為該索引選擇度較差,就有可能導(dǎo)致使用錯(cuò)誤的索引。
三、統(tǒng)計(jì)信息不準(zhǔn)確的處理
我們查看執(zhí)行計(jì)劃,發(fā)現(xiàn)未使用正確的索引,如果是innodb_index_stats中統(tǒng)計(jì)信息差別較大引起,可通過(guò)以下方式處理:
1、手動(dòng)更新統(tǒng)計(jì)信息,注意執(zhí)行過(guò)程中會(huì)加讀鎖:
ANALYZETABLE TABLE_NAME;
2、如果更新后統(tǒng)計(jì)信息仍不準(zhǔn)確,可考慮增加表采樣的數(shù)據(jù)頁(yè),兩種方式可以修改:
a) 全局變量INNODB_STATS_PERSISTENT_SAMPLE_PAGES,默認(rèn)為20;
b) 單個(gè)表可以指定該表的采樣:
ALTER TABLE TABLE_NAME STATS_SAMPLE_PAGES=40;
經(jīng)測(cè)試,此處STATS_SAMPLE_PAGES的最大值是65535,超出會(huì)報(bào)錯(cuò)。
目前MySQL并沒(méi)有提供直方圖的功能,某些情況下(如數(shù)據(jù)分布不均)僅僅更新統(tǒng)計(jì)信息不一定能得到準(zhǔn)確的執(zhí)行計(jì)劃,只能通過(guò)index hint的方式指定索引。新版本8.0會(huì)增加直方圖功能,讓我們期待MySQL越來(lái)越強(qiáng)大的功能吧!
相關(guān)文章
Mysql中正則表達(dá)式Regexp常見(jiàn)用法
這篇文章主要介紹了Mysql中正則表達(dá)式Regexp常見(jiàn)用法,MySql REGEXP運(yùn)算符匹配字符串,mysql正則REGEXP學(xué)習(xí)練習(xí)筆記,需要的朋友可以參考下2020-02-02MySQL修改默認(rèn)存儲(chǔ)引擎的實(shí)現(xiàn)方法
下面小編就為大家?guī)?lái)一篇MySQL修改默認(rèn)存儲(chǔ)引擎的實(shí)現(xiàn)方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-03-03MySQL如何優(yōu)雅的備份賬號(hào)相關(guān)信息
這篇文章主要介紹了MySQL如何優(yōu)雅的備份賬號(hào)相關(guān)信息,幫助大家更好的理解和學(xué)習(xí)MySQL,感興趣的朋友可以了解下2020-08-08Mysql表數(shù)據(jù)比較大情況下修改添加字段的方法實(shí)例
最近遇到的一個(gè)問(wèn)題,需要在一張1800萬(wàn)數(shù)據(jù)量的表中添加加一個(gè)字段,所以這篇文章主要給大家介紹了關(guān)于Mysql表數(shù)據(jù)比較大情況下修改添加字段的方法,需要的朋友可以參考下2022-06-06mySQL中LEN()與DATALENGTH()的區(qū)別
LEN返回指定字符串表達(dá)式的字符數(shù),其中不包含尾隨空格。DATALENGTH返回用于表示任何表達(dá)式的字節(jié)數(shù)。2011-03-03mysql啟動(dòng)服務(wù)報(bào)1058錯(cuò)誤的解決方法
這篇文章主要介紹了mysql啟動(dòng)服務(wù)報(bào)1058錯(cuò)誤的解決方法,需要的朋友可以參考下2014-03-03MySQL 5.6下table_open_cache參數(shù)優(yōu)化合理配置詳解
這篇文章主要介紹了MySQL 5.6下table_open_cache參數(shù)合理配置詳解,需要的朋友可以參考下2018-03-03重置MySQL中表中自增列的初始值的實(shí)現(xiàn)方法
這篇文章主要介紹了重置MySQL中表中自增列的初始值的實(shí)現(xiàn)方法的相關(guān)資料,需要的朋友可以參考下2017-03-03Mysql如何導(dǎo)出篩選數(shù)據(jù)并導(dǎo)出帶表頭的csv文件
這篇文章主要介紹了Mysql如何導(dǎo)出篩選數(shù)據(jù)并導(dǎo)出帶表頭的csv文件,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07