您好,登錄后才能下訂單哦!
今天小編給大家分享一下MySQL不適合構建索引及索引失效的情況有哪些的相關知識點,內容詳細,邏輯清晰,相信大部分人都還太了解這方面的知識,所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。
具體案例下文有詳盡描述
不適合建立索引的場景:
數據量比較小的表不建議建立索引
有大量重復數據的字段上不建議建立索引(類似:性別字段)
需要進行頻繁更新的表不建議建立索引
where、group by、order by后面的沒有使用到的字段不建立索引
不要定義冗余索引
索引失效的場景:
過濾條件使用不等于(!=、<>)
過濾條件使用is not null
在索引字段上使用函數或進行計算
在使用聯合索引的時候,需要滿足“最佳左前綴法則”,否則失效
當使用了類型轉換也會導致索引失效
在使用范圍查詢的時候,聯合索引的部分字段失效(where age >18)
在like字段中,如果是以%開頭,索引失效(where name like ‘%abc’)
在使用or進行查詢的時候,or前后出現非索引字段,索引失效
表和庫的字符集不一致,回導致索引失效
知識點:
每張表的索引不建議超過6個(占用空間、降低表更新速度)
最終到底是否使用索引還是優化器進行決定的
優化器會根據數據量、數據庫版本、數據選擇讀進行查詢代價的比較,從而決定是否使用索引
建立索引的時候將需要范圍匹配的字段建立在索引的尾部,避免失效
在建立表的時候將字段設置為not null同時設置默認值,當需要查找沒有值的記錄的時候就可以使用where xxx = 默認值,放置使用is not null導致索引失效
頁面搜索的時候嚴謹左模糊或者全模糊(like ‘%abc’)
對于過濾性較好的字段建立在聯合索引的前面,這樣就可以優先過濾比較多的數據
場景一:數據少的表
當數據比較少的時候,索引的優勢就不明顯了,因為數據庫的存儲引擎也是非常快的,相較于需要查詢索引在進行回表操作,可能直接查詢的性能會更高一些,所以數據相對較少的表不建議建立索引
場景二:有大量重復數據的字段
類似于性別字段,只有“男”和“女”兩個不同的值,所以索引一半的數據是“男”一半的數據是“女”,那么建立索引并不能進行快速的查詢等,所以不建議在有大量重復數據的列上建立索引
場景三:頻繁更新的表(update/delete/insert)
因為表中更新數據的時候,索引也是需要進行對應的維護的,如果一個表近期需要頻繁的進行增刪改操作,那么就需要耗費大量的時間去維護索引,不建議建立索引,可以在需要進行頻繁的更新操作的時候將索引刪除,更新完畢之后重建索引
場景四:沒有使用的字段(where/group by/order by)
不是where/group by/order by后面的字段沒有必要建立索引,因為不會使用到該索引
場景五:不要定義冗余索引
create index username_password_address on xiao(username,password,address); -- 如果建立了第一個索引,那么就沒有必要建立第二個索引 create index username on xiao (username); --第二個索引就是冗余索引,因為第一個已經是先根據username排序的索引 --也就是第二個索引的功能完全可以由第一個索引實現
這里因為username作為第一個聯合索引的第一個字段,所以索引就是按照username進行排序,在username相同的情況下按照password、address排序,所以也就是實現了單獨拿username列作為索引的功能,即第二個索引就是多余的
場景一:在建立索引的字段上進行運算(函數等),導致索引失效
這里首先是給age創建了索引,在第一次查詢過程中使用了age索引,但是第二次key值為null(索引失效),導致索引失效的原因在于第二次查詢的時候where后面對age進行了計算,計算機并不知道執行的是什么計算所以會將age+1計算后與1比較,索引失效
類似于在字段上使用函數concat()等都會導致索引失效
場景二:使用不等于(where age != 18)
當使用等值運算,那么是可以在索引中進行查找的,但是如果是不等于,那么則需要遍歷所有數據,所以所失效
explain select * from xiaoyuanhao where age = 18; explain select * from xiaoyuanhao where age != 18; --這里是在age字段上建立了普通索引,第二個查詢時候索引失效
場景三:使用is not null索引失效
與不等于一樣,如果使用的是is not null,那么就需要進行全部數據的遍歷操作,索引失效,但是如果使用的是is null那么依舊是可以使用索引的
--這里是在age字段上建立了普通索引,第二個查詢時候索引失效 explain select * from xiaoyuanhao where age is null; --可以正常使用索引 explain select * from xiaoyuanhao where age is not null; --索引失效
場景四:在使用聯合索引的時候沒有遵循最佳左前綴法則
CREATE INDEX age_classid_name ON student(age,classId,NAME); EXPLAIN SELECT * FROM student WHERE classId = 30 AND NAME = 'xiaoyuanhao'; -- 因為沒有使用age字段,所以沒有準許最佳左前綴原則,索引失效
從這里可以看出是沒有使用索引的(key = null),因為創建的索引是先按照age進行排序,在age相同的情況下按照classId和name排序,如果在查詢的時候需要直接按照classId進行排序查找,那么就無法使用該索引,即索引失效。
如果需要使用使用索引,那么就一定需要到聯合索引的第一個字段age,案例如下
EXPLAIN SELECT * FROM student WHERE age = 10 AND NAME = 'xiaoyuanhao'; EXPLAIN SELECT * FROM student WHERE age = 10 AND classId = 33 AND NAME = 'xiaoyuanhao'; --兩者都是使用age字段索引,所以索引有效
場景五:類型轉換導致索引失效
CREATE INDEX NAME ON student(NAME); -- 這里的name字段是varchar類型 EXPLAIN SELECT * FROM student WHERE NAME = 'xiao'; -- 本次查詢是可以使用索引的,因為類型都是一致的,都是字符串 EXPLAIN SELECT * FROM student WHERE NAME = 123; -- 本次查詢則無法使用索引,因為是將數字類型123轉換為字符類型
沒有發生類型轉換,使用索引key = name
發生了類型轉換,無法使用索引kye = null,索引失效
使用索引的時候一定需要保證數據類型是一致的,否則系統就需要進行轉換,那么就無法使用索引
場景六:使用范圍查詢導致聯合索引其他字段失效
create index age_classId_name on student (age,classId,name); EXPLAIN SELECT * FROM student WHERE age = 10 AND classId > 20 AND NAME = 'xiaoyuanhao'; -- 這里只能使用age,classId,索引的前兩個字段 EXPLAIN SELECT * FROM student WHERE age = 10 AND classId = 20 AND NAME = 'xiaoyuanhao'; -- 這里可以使用完整的索引,因為都是等值連接
在classId字段上使用范圍查詢,導致name字段失效,有效索引長度為63
使用的都是等值匹配,整個索引皆可用,有效索引長度為73
也就是在對于聯合索引來說,如果在使用的時候是等值匹配,那么就可以重復的利用索引,如果不是等值匹配,那么該字段也是可以使用索引的,但是該字段右邊的字段就將失效
建議在建立索引的時候將需要范圍匹配的字段建立在索引的最后面
場景七:在使用like的時候,如果以%開頭導致索引失效
EXPLAIN SELECT * FROM student WHERE NAME LIKE 'abc%'; -- 可以正常使用索引 EXPLAIN SELECT * FROM student WHERE NAME LIKE '%abc'; -- 這里在like中,%在前面無法使用索引
key = name,使用了該索引,索引有效
key = null,索引失效
因為建立的索引實際上是按照整個字符串的從第一個開始進行比較排序的,所以在使用like的時候,也只能夠重現進行比較,如果使用的是’%abc’,那么查詢的就是以abc結尾的數據,無法使用索引
場景八:or前后出現非索引字段,索引失效
-- 該表中只有name字段上的索引 CREATE INDEX NAME ON student(NAME); EXPLAIN SELECT * FROM student WHERE NAME = 'xiao'; -- 這里是可以使用name索引的 EXPLAIN SELECT * FROM student WHERE NAME = 'xiao' OR classId = 1001; -- 這個則無法使用索引,進行的是全表掃描
key = null,無法使用索引,or條件中出現非索引字段
因為如果name不等于’xiao’的時候那么就會繼續判斷classId是否等于1001,那么實際上還是會進行全表掃描,所以索引失效(也就是進行name判斷的時候可以使用索引,但是在判斷classId的時候又要全表掃描,那么優化器就直接進行全表掃描),但是如果or前后的字段都有索引了,那么就就會使用索引
以上就是“MySQL不適合構建索引及索引失效的情況有哪些”這篇文章的所有內容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學習更多的知識,請關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。