【索引】聚合索引


1. 普通青年的索引使用方式

假設我們有一個用戶表 tb_user,內容如下:

name age sex
jack 22
rose 21
tom 20
... ... ...

執行SQL語句:

SELECT name FROM tb_user WHERE age = 20;

默認情況下,MySQL需要遍歷整張表,才能找到符合條件的記錄。如果在age字段上建立索引,那么MySQL可以很快找到所有符合條件的記錄(索引本身通過B+樹實現,查起來很快。簡單起見,想象一下二分查找和遍歷查找的區別。)

2. 文藝青年的玩法

2.1 用冗余的聯合索引加速查詢

接着上面的例子,我們假設,tb_user表有一百萬行,通常情況下,"WHERE age = 20"這樣的語句,會返回幾萬行數據,實際測試下發現,速度不夠快。

原因是,MySQL根據索引查詢到符合條件的記錄后,還需要到表空間里一一查找這些記錄(實際上,索引里同時記錄了age字段和關聯記錄的物理行號),這意味着,MySQL必須讀取表空間多達幾萬次,才能返回最終結果。

聰明的你可能已經想到了,如果age字段的索引上有name字段的值話,MySQL就不用再費事地去訪問表空間了。

最終解決方案:建立聯合索引,讓MySQL直接從索引中取出name字段的值

KEY `age_with_name` (`age`,`name`)

注意這里的順序,必須是先age后name,反之不行(除非你是根據name查age)。

2.2 用冗余的聯合索引加速排序

依然是之前的表,假設要做這樣的查詢:

SELECT * FROM tb_user ORDER BY age;

因為我們在age上有索引,所以排序是很快的(索引的本質就是將表記錄的物理行號按照特定規則排序)

實際項目中,SQL可能比這個復雜些,比如:

SELECT * FROM tb_user WHERE sex='' ORDER BY age;

這個時候,age字段上的索引就派不上用場了。因為,age索引是面向整個表的,篩選后的表和age索引是對不上的。

解決方案:依然是聯合索引!

KEY `age_with_name` (`sex`,`age`)

這個聯合索引,同時記錄了sex和age,並且排序的規則是,先按sex排,sex相同時按age排。那么,通過"WHERE sex='男'",MySQL先對索引進行篩選,然后剩下的索引正好就是按照age排序的了。因此,整個SQL的排序速度依然很快。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM