用一個(gè)例子詳細(xì)說(shuō)明單列索引與組合索引的區(qū)別及索引使用中的一些細(xì)節(jié)
建一個(gè)表:
CREATE TABLE myIndex (
i_testID INT NOT NULL AUTO_INCREMENT,
vc_Name VARCHAR(50) NOT NULL,
vc_City VARCHAR(50) NOT NULL,
i_Age INT NOT NULL,
i_SchoolID INT NOT NULL,
PRIMARY KEY (i_testID)
);
假設(shè)這個(gè)表中有10000條記錄,其中很分散地分布了5條vc_Name="erquan"的記錄,只不過(guò)city,age,school的組合各不相同。
來(lái)看這條T-SQL:
SELECT i_testID FROM myIndex WHERE vc_Name='erquan' AND vc_City='beijing' AND i_Age=25;
首先考慮建單列索引:
在vc_Name列上建立了索引。執(zhí)行T-SQL時(shí),MYSQL很快將目標(biāo)鎖定在了vc_Name=erquan的5條記錄上,取出來(lái)放到一中間 結(jié)果集。在這個(gè)結(jié)果集里,先排除掉vc_City不等于"beijing"的記錄,再排除i_Age不等于25的記錄,最后篩選出唯一的符合條件的記錄。
雖然在vc_Name上建立了索引,查詢時(shí)MYSQL不用掃描整張表,效率有所提高,但離我們的要求還有一定的距離。同樣的,在vc_City和i_Age分別建立的單列索引的效率相似。
為了進(jìn)一步榨取MySQL的效率,就要考慮建立組合索引。就是將vc_Name,vc_City,i_Age建到一個(gè)索引里:
ALTER TABLE myIndex ADD INDEX name_city_age (vc_Name(10),vc_City,i_Age);--注意了,建表時(shí),vc_Name長(zhǎng)度為50,這里為什么用10呢?因?yàn)橐话闱闆r下名字的長(zhǎng) 度不會(huì)超過(guò)10,這樣會(huì)加速索引查詢速度,還會(huì)減少索引文件的大小,提高INSERT的更新速度。
執(zhí)行T-SQL時(shí),MySQL無(wú)須掃描任何記錄就到找到唯一的記錄!
肯定有人要問(wèn)了,如果分別在vc_Name,vc_City,i_Age上建立單列索引,讓該表有3個(gè)單列索引,查詢時(shí)和上述的組合索引效率一樣 吧?嘿嘿,大不一樣,遠(yuǎn)遠(yuǎn)低于我們的組合索引~~雖然此時(shí)有了三個(gè)索引,但MySQL只能用到其中的那個(gè)它認(rèn)為似乎是最有效率的單列索引。
建立這樣的組合索引,其實(shí)是相當(dāng)于分別建立了
vc_Name,vc_City,i_Age
vc_Name,vc_City
vc_Name
這樣的三個(gè)組合索引為什么沒(méi)有vc_City,i_Age等這樣的組合索引的效率呢?這是因?yàn)?/span>mysql組合索引"最左前綴"的結(jié)果。簡(jiǎn)單的理解就是只從最左面的開(kāi)始組合。并不是只要包含這三列的查詢都會(huì)用到該組合索引,下面的幾個(gè)T-SQL會(huì)用到:
SELECT * FROM myIndex WHREE vc_Name="erquan" AND vc_City="beijing"
SELECT * FROM myIndex WHREE vc_Name="erquan"
而下面幾個(gè)則不會(huì)用到:
SELECT * FROM myIndex WHREE i_Age=20 AND vc_City="beijing"
SELECT * FROM myIndex WHREE vc_City="beijing"
但什么情況下需要建立索引呢?一般來(lái)說(shuō),在WHERE和JOIN中出現(xiàn)的列需要建立索引,但也不完全如此,因?yàn)镸ySQL只對(duì) <,<=,=,>,>=,BETWEEN,IN,以及某些時(shí)候的LIKE才會(huì)使用索引。
SELECT t.vc_Name FROM testIndex t LEFT JOIN myIndex m ON t.vc_Name=m.vc_Name WHERE m.i_Age=20 AND m.vc_City='beijing' 時(shí),有對(duì)myIndex表的vc_City和i_Age建立索引的需要,由于testIndex表的vc_Name開(kāi)出現(xiàn)在了JOIN子句中,也有對(duì)它建立索引的必要。
剛才提到只有某些時(shí)候的LIKE才需建立索引?是的。因?yàn)樵谝酝ㄅ浞?% 和 _ 開(kāi)頭作查詢時(shí),MySQL不會(huì)使用索引,如
SELECT * FROM myIndex WHERE vc_Name like'erquan%'
會(huì)使用索引,而
SELECT * FROM myIndex WHEREt vc_Name like'%erquan'
就不會(huì)使用索引了。
posted @ 2007-09-22 23:19 擦肩而過(guò) 閱讀(308) | 評(píng)論 (0) | 編輯 收藏