Mysql中between...and引起的索引失效問(wèn)題及解決
發(fā)生場(chǎng)景
在查詢學(xué)生表的時(shí)候,需要支持根據(jù)創(chuàng)建時(shí)間來(lái)篩選出某段時(shí)間內(nèi)入學(xué)的學(xué)生總數(shù),因此在創(chuàng)建時(shí)間上加了索引,但是最終發(fā)現(xiàn)還是會(huì)走全量查詢。
實(shí)驗(yàn)過(guò)程
CREATE TABLE `t_user` ( `id` bigint(11) unsigned NOT NULL COMMENT '學(xué)生id', `name` varchar(24) NOT NULL COMMENT '學(xué)生名稱', `createTime` dat NOT NULL COMMENT '創(chuàng)建時(shí)間', PRIMARY KEY (`id`), KEY `index_updateTime` (`createTime`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4
select count(1) from t_user where createTime between '2022-08-30' and '2022-08-31';
使用insert語(yǔ)句插入3萬(wàn)多條數(shù)據(jù)后,其中30號(hào)至31號(hào)之前有9千多條數(shù)據(jù),發(fā)現(xiàn)查詢的時(shí)候并沒(méi)有走到索引。
結(jié)果如下:
縮小查詢區(qū)間,只查31號(hào)的(4千多條數(shù)據(jù)),卻發(fā)現(xiàn)可以走到索引了,
結(jié)果如下:
select count(1) from t_user where createTime between '2022-08-31' and '2022-08-31';
再找一個(gè)數(shù)據(jù)量為5千的區(qū)間來(lái)試一下,也是會(huì)走到索引的,
結(jié)果如下:
select count(1) from t_user where createTime between '2022-08-23' and '2022-08-25';
沒(méi)走到索引是因?yàn)閎etween…and引起的嗎?如果改為>=和<=呢。
結(jié)果如下:
select count(1) from t_user where createTime >= '2022-08-30' and createTime <= '2022-08-31';
結(jié)論
經(jīng)過(guò)實(shí)驗(yàn)發(fā)現(xiàn),當(dāng)查詢的數(shù)據(jù)量達(dá)到6千(占比20%左右),就不走索引了。
引起原因:
表的數(shù)據(jù)量太大,會(huì)讓數(shù)據(jù)庫(kù)中的優(yōu)化器進(jìn)行處理。
優(yōu)化器是在表里面有多個(gè)索引的時(shí)候,決定使用哪個(gè)索引,查詢的量太大,導(dǎo)致優(yōu)化器認(rèn)為走全表查詢時(shí)間效率更佳。
但是如果一定要用到區(qū)間查詢,這個(gè)問(wèn)題該如何解決呢?
方案一
select count(1) from t_user where createTime between '2022-08-30' and '2022-08-31' limit 1;
方案二
select count(1) from t_user FORCE INDEX(index_updateTime) where createTime between '2022-08-30' and '2022-08-31';
思考:
當(dāng)查詢數(shù)據(jù)量達(dá)到一定量的時(shí)候會(huì)導(dǎo)致between…and索引失效,那分頁(yè)查詢的時(shí)候呢?
select id,name, createTime from t_user where createTime between '2022-08-30' and '2022-08-31' limit 1,500;
可見(jiàn),查詢第一頁(yè)的時(shí)候索引有效,但是隨著頁(yè)碼越來(lái)越大的時(shí)候,索引卻失效了,
select id,name, createTime from t_user where createTime between '2022-08-30' and '2022-08-31' limit 100000,500;
因此,遇到這種因查詢數(shù)據(jù)量過(guò)大而導(dǎo)致的索引失效的問(wèn)題,需要對(duì)其功能做相應(yīng)限制處理。
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
mysql中復(fù)制表結(jié)構(gòu)的方法小結(jié)
這篇文章主要介紹了mysql中復(fù)制表結(jié)構(gòu)的方法,需要的朋友可以參考下2014-07-07mysql導(dǎo)入csv的4種報(bào)錯(cuò)的解決方法
這篇文章主要介紹了mysql導(dǎo)入csv的4種報(bào)錯(cuò)的解決方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12mysql實(shí)現(xiàn)sequence功能的代碼
今天小編就為大家分享一篇關(guān)于mysql實(shí)現(xiàn)sequence功能的代碼,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2019-03-03Mysql 導(dǎo)入導(dǎo)出csv 中文亂碼問(wèn)題的解決方法
這篇文章介紹了Mysql 導(dǎo)入導(dǎo)出csv 中文亂碼問(wèn)題的解決方法,有需要的朋友可以參考一下2013-09-09Mysql中where與on的區(qū)別及何時(shí)使用詳析
MySQL當(dāng)中的限制條件可以使用on或者where,兩者在不同的情況下具有不同而意義,這篇文章主要給大家介紹了關(guān)于Mysql中where與on的區(qū)別及何時(shí)使用的相關(guān)資料,需要的朋友可以參考下2021-08-08MySQL創(chuàng)建數(shù)據(jù)表時(shí)設(shè)定引擎MyISAM/InnoDB操作
這篇文章主要介紹了MySQL創(chuàng)建數(shù)據(jù)表時(shí)設(shè)定引擎MyISAM/InnoDB操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-08-08解決創(chuàng)建主鍵報(bào)錯(cuò):Incorrect column specifier for
這篇文章主要介紹了解決創(chuàng)建主鍵報(bào)錯(cuò):Incorrect column specifier for column‘id‘問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-08-08MySQL中使用replace、regexp進(jìn)行正則表達(dá)式替換的用法分析
這篇文章主要介紹了MySQL中使用replace、regexp進(jìn)行正則表達(dá)式替換的用法,結(jié)合具體實(shí)例形式分析了replace、regexp正則替換的使用技巧與相關(guān)注意事項(xiàng),需要的朋友可以參考下2017-03-03mysql如何按字段查詢重復(fù)的數(shù)據(jù)
這篇文章主要介紹了mysql如何按字段查詢重復(fù)的數(shù)據(jù)問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-05-05