MySQL中索引優(yōu)化distinct語(yǔ)句及distinct的多字段操作
MySQL通常使用GROUPBY(本質(zhì)上是排序動(dòng)作)完成DISTINCT操作,如果DISTINCT操作和ORDERBY操作組合使用,通常會(huì)用到臨時(shí)表.這樣會(huì)影響性能. 在一些情況下,MySQL可以使用索引優(yōu)化DISTINCT操作,但需要活學(xué)活用.本文涉及一個(gè)不能利用索引完成DISTINCT操作的實(shí)例.
實(shí)例1 使用索引優(yōu)化DISTINCT操作
create table m11 (a int, b int, c int, d int, primary key(a)) engine=INNODB; insert into m11 values (1,1,1,1),(2,2,2,2),(3,3,3,3),(4,4,4,4),(5,5,5,5),(6,6,6,6),(7,7,7,7),(8,8,8,8); explain select distinct(a) from m11;
mysql> explain select distinct(a) from m11;
+----+-------------+-------+------------+-------+---------------+---------+---------+------+------+----------+-------------+| id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra |+----+-------------+-------+------------+-------+---------------+---------+---------+------+------+----------+-------------+| 1 | SIMPLE | m11 | NULL | index | PRIMARY | PRIMARY | 4 | NULL | 1 | 100.00 | Using index |+----+-------------+-------+------------+-------+---------------+---------+---------+------+------+----------+-------------+
說(shuō)明:
1 'a'列上存在主鍵索引,MySQL可以利用索引(key列值表明使用了主鍵索引)完成了DISTINCT操作.
2 這是使用索引優(yōu)化DISTINCT操作的典型實(shí)例.
實(shí)例2 使用索引不能優(yōu)化DISTINCT操作
create table m31 (a int, b int, c int, d int, primary key(a)) engine=MEMORY; insert into m31 values (1,1,1,1),(2,2,2,2),(3,3,3,3),(4,4,4,4),(5,5,5,5),(6,6,6,6),(7,7,7,7),(8,8,8,8); explain select distinct(a) from m31;
mysql> explain select distinct(a) from m31;
+----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------+| id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra |+----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------+| 1 | SIMPLE | m31 | NULL | ALL | NULL | NULL | NULL | NULL | 8 | 100.00 | NULL |+----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------+
說(shuō)明:
1 從查詢執(zhí)行計(jì)劃看,索引沒(méi)有被使用.
2 對(duì)比實(shí)例1的建表語(yǔ)句,只是存儲(chǔ)引擎不同.
3 為什么主鍵索引沒(méi)有起作用? 難道MEMORY存儲(chǔ)引擎上的索引不可使用?
實(shí)例3 使用索引可以優(yōu)化DISTINCT操作的Memory表
create table m33 (a int, b int, c int, d int, INDEX USING BTREE (a)) engine=MEMORY; insert into m33 values (1,1,1,1),(2,2,2,2),(3,3,3,3),(4,4,4,4),(5,5,5,5),(6,6,6,6),(7,7,7,7),(8,8,8,8); explain select distinct(a) from m33;
mysql> explain select distinct(a) from m33;
+----+-------------+-------+------------+-------+---------------+------+---------+------+------+----------+-------+| id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra |+----+-------------+-------+------------+-------+---------------+------+---------+------+------+----------+-------+| 1 | SIMPLE | m33 | NULL | index | NULL | a | 5 | NULL | 8 | 100.00 | NULL |+----+-------------+-------+------------+-------+---------------+------+---------+------+------+----------+-------+
說(shuō)明:
1 'a'列上存在主鍵索引,MySQL可以利用索引(key列值表明使用了主鍵索引)完成了DISTINCT操作.
2 對(duì)比實(shí)例2,可以發(fā)現(xiàn),二者都使用了Memory引擎. 但實(shí)例3指名使用Btree類型的索引.
3 實(shí)例2沒(méi)有指定使用什么類型的索引,MySQL將采用默認(rèn)值. MySQL手冊(cè)上說(shuō):
As indicated by the engine name, MEMORY tables are stored in memory. They use hash indexes by default, which makes them very fast for single-value lookups, and very useful for creating temporary tables.
結(jié)論:
1 看索引對(duì)查詢的影響,要注意索引的類型.
2 HASH索引適合等值查找,但不適合需要有序的場(chǎng)景,而Btree卻適合有序的場(chǎng)景.
3 看查詢執(zhí)行計(jì)劃,發(fā)現(xiàn)索引沒(méi)有被使用,需要進(jìn)一步考察索引的類型.
DISTINCT不能選擇多個(gè)字段的解決方法
在實(shí)際應(yīng)用中,我們經(jīng)常要選擇數(shù)據(jù)庫(kù)某表中重復(fù)數(shù)據(jù),通常我們是使用DISTINCT函數(shù)。
但DISTINCT只能對(duì)一個(gè)字段有效,比如:
sql="select DISTINCT title from Table where id>0"
當(dāng)我們需要列出數(shù)據(jù)中的另一列,比如:
sql="select DISTINCT title,posttime from Table where id>0"
得出的結(jié)果就不是我們想要的了,所以我們需要用另外的方法來(lái)解決這個(gè)問(wèn)題。
下面的是我寫的SQL語(yǔ)句,我不知道是不是很好,但愿有更好的人拿出來(lái)分享一下:
寫法一:
sql = "Select DISTINCT(title),posttime From Table1 Where id>0"
寫法二:
sql = "Select title,posttime From Table1 Where id>0 group by title,posttime"
寫法三:
sql="select title,posttime from Table where id in (select min(id) from Table group by title)"
相關(guān)文章
MySQL 那些常見(jiàn)的錯(cuò)誤設(shè)計(jì)規(guī)范,你都知道嗎
今天來(lái)看一看 MySQL 設(shè)計(jì)規(guī)范中幾個(gè)常見(jiàn)的錯(cuò)誤例子,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2021-07-07在MySQL中使用JOIN語(yǔ)句進(jìn)行連接操作的詳細(xì)教程
這篇文章主要介紹了在MySQL中使用JOIN語(yǔ)句進(jìn)行連接操作的詳細(xì)教程,是MySQL入門學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下2015-05-05navicat 8 創(chuàng)建數(shù)據(jù)庫(kù)與創(chuàng)建用戶分配權(quán)限圖文方法
navicat是一款不錯(cuò)的圖形化管理mysql的工具,大家一般都是用phpmyadmin或直接命令行操作,對(duì)于不是很熟悉命令的朋友,就可以使用navicat這個(gè)工具了,方便操作。2011-04-04mysql實(shí)現(xiàn)按照某個(gè)時(shí)間段分組統(tǒng)計(jì)
這篇文章主要介紹了mysql實(shí)現(xiàn)按照某個(gè)時(shí)間段分組統(tǒng)計(jì)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-10-10基于MySQL的存儲(chǔ)引擎與日志說(shuō)明(全面講解)
下面小編就為大家分享一篇基于MySQL的存儲(chǔ)引擎與日志說(shuō)明(全面講解),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2017-12-12MySql5.7.11編譯安裝及修改root密碼的方法小結(jié)
這篇文章主要介紹了MySql5.7.11編譯安裝及修改root密碼的方法小結(jié)的相關(guān)資料,需要的朋友可以參考下2016-04-04淺談Mysql、SqlServer、Oracle三大數(shù)據(jù)庫(kù)的區(qū)別
這篇文章主要介紹了Mysql、SqlServer、Oracle三大數(shù)據(jù)庫(kù)的區(qū)別,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04