MySQL優(yōu)化案例之隱式字符編碼轉(zhuǎn)換
索性失效前提
MySQL中我們知道有:
- 1、如果對(duì)索引字段做函數(shù)操作,可能會(huì)破壞索引值的有序性,因此優(yōu)化器就決定放棄走樹(shù)搜索功能。
- 2、隱式類(lèi)型轉(zhuǎn)換也會(huì)導(dǎo)致同樣的放棄走樹(shù)搜索。
因?yàn)轭?lèi)型轉(zhuǎn)換等價(jià)于在條件字段上使用了函數(shù)比如:
/*假設(shè)tradeid字段有索引,且為varchar類(lèi)型*/ mysql> select * from tradelog where tradeid=110717; /*等價(jià)于*/ mysql> select * from tradelog where CAST(tradid AS signed int) = 110717;
一個(gè)真實(shí)的案例
下面來(lái)看看隱式字符編碼轉(zhuǎn)換導(dǎo)致的一個(gè)慢sql
優(yōu)化前原始sql分析
業(yè)務(wù)上有個(gè)sql執(zhí)行需要1.31秒
看看執(zhí)行計(jì)劃:
從執(zhí)行計(jì)劃分析看出問(wèn)題出在r表也就是 h_merge_result_new_indicator 表全表掃描,查看該表的表結(jié)構(gòu)有聯(lián)合索引。但是聯(lián)合索引范圍后會(huì)失效,于是打算新建一個(gè)聯(lián)合索引。
優(yōu)化初步處理
查看預(yù)新建聯(lián)合索引的字段選擇性:
結(jié)合選擇性來(lái)看;
create index idx_hmrni on h_merge_result_new_indicator(keyName,module,BATCH_NO);
初步優(yōu)化無(wú)效分析
創(chuàng)建后,再次查看執(zhí)行計(jì)劃依然無(wú)效;
查看表結(jié)構(gòu):
另外3個(gè)表結(jié)構(gòu)其中有2個(gè)utf8mb4,1個(gè)utf8
字符集 utf8mb4 是 utf8 的超集,所以當(dāng)這兩個(gè)類(lèi)型的字符串在做比較的時(shí)候,MySQL 內(nèi)部的操作是,先把 utf8 字符串轉(zhuǎn)成 utf8mb4 字符集,再做比較。
因此:
這部分會(huì)轉(zhuǎn)換后再與h_merge_result_new_indicator關(guān)聯(lián)
第二次優(yōu)化處理
優(yōu)化就只需要將字符集編碼轉(zhuǎn)為utf8再和h_merge_result_new_indicator關(guān)聯(lián)就能用上索引
再看查詢(xún)只需要0.02秒了
第三次優(yōu)化
但是還有個(gè)問(wèn)題,如上執(zhí)行計(jì)劃key_len是606 =(100*3+3)+(100*3+3)
也就是說(shuō),沒(méi)有用上BATCH_NO字段上的索引,我們知道索引少一個(gè)字段,占用會(huì)減少,不會(huì)太臃腫,因此,聯(lián)合索引只需要包含r(keyName,module)
- drop index idx_hmrni on h_merge_result_new_indicator;
- create index idx_hmrni on h_merge_result_new_indicator(keyName,module);
結(jié)論
對(duì)索引字段做函數(shù)操作,可能會(huì)破壞索引值的有序性,因此優(yōu)化器就決定放棄走樹(shù)搜索功能。該例子是隱式字符編碼轉(zhuǎn)換,它們都跟其他條件索引上使用函數(shù)一樣,因?yàn)橐笤谒饕侄紊献龊瘮?shù)操作而導(dǎo)致了全索引掃描。
MySQL 的優(yōu)化器確實(shí)有“偷懶”的嫌疑,即使簡(jiǎn)單地把 where id+1=1000 改寫(xiě)成 where id=1000-1 就能夠用上索引快速查找,也不會(huì)主動(dòng)做這個(gè)語(yǔ)句重寫(xiě)。
保證在條件索引上不做破壞索引值的有序性,是優(yōu)化索引的利器。
到此這篇關(guān)于MySQL優(yōu)化案例之隱式字符編碼轉(zhuǎn)換的文章就介紹到這了,更多相關(guān)MySQL隱式字符編碼轉(zhuǎn)換內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MySQL學(xué)習(xí)(七):Innodb存儲(chǔ)引擎索引的實(shí)現(xiàn)原理詳解
這篇文章主要介紹了Innodb存儲(chǔ)引擎索引的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04mysql使用教程之分區(qū)表的使用方法(刪除分區(qū)表)
mysql分區(qū)表使用方法,新增分區(qū)、刪除分區(qū)、分區(qū)的合并、分區(qū)的拆分等使用方法2013-12-12sql語(yǔ)句中l(wèi)ike的用法詳細(xì)解析
以下是對(duì)sql語(yǔ)句中l(wèi)ike的用法進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過(guò)來(lái)參考下2013-08-08MySQL存儲(chǔ)過(guò)程的查詢(xún)命令介紹
這篇文章主要介紹了MySQL存儲(chǔ)過(guò)程的查詢(xún)命令介紹,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-02-02mysql left join快速轉(zhuǎn)inner join的過(guò)程
inner join內(nèi)連接,顯示兩個(gè)表中有聯(lián)系的所有數(shù)據(jù)而left join,左鏈接,以左表為參照,顯示所有數(shù)據(jù),右表中沒(méi)有則以null顯示,本文重點(diǎn)給大家講解mysql left join快速轉(zhuǎn)inner join的過(guò)程,需要的朋友參考下吧2021-06-06MySQL 數(shù)據(jù)庫(kù)函數(shù)庫(kù)
MySQL 數(shù)據(jù)庫(kù)函數(shù)庫(kù)...2006-12-12mysql 常見(jiàn)命令和學(xué)習(xí)心得
mysql 常見(jiàn)命令和學(xué)習(xí)心得,學(xué)習(xí)php的朋友需要了解的一些基礎(chǔ),方便日后的開(kāi)發(fā)。2009-09-09MySQL?數(shù)據(jù)庫(kù)聚合查詢(xún)和聯(lián)合查詢(xún)操作
這篇文章主要介紹了MySQL?數(shù)據(jù)庫(kù)聚合查詢(xún)和聯(lián)合查詢(xún)操作,需要的朋友可以參考下2021-12-12