搜尋
首頁資料庫mysql教程mysql索引詳解(總結)

mysql索引詳解(總結)

Jan 31, 2020 pm 05:38 PM
mysql

mysql索引詳解(總結)

上文《關於mysql 執行流程的解析》中我們主要介紹了sql語句在server層的執行過程

我們再來分析一下具體的語句在引擎層的執行步驟,CRUD的操作都跟索引相關,我們先了解一下索引

索引

##索引的出現其實就是為了提高資料查詢的效率,就像書的目錄

資料結構

常見的資料結構有雜湊表、有序數組和搜尋樹

#雜湊表是一種以鍵- 值(key-value)儲存資料的結構,我們只要輸入待查找的值即key, 就可以找到其對應的值即Value。哈希的想法很簡單,把值放在數組裡,用一個哈希函數把key 換算成一個位置,然後把value 放在數組的對應位置

不可避免地,多個key 值經過哈希函數的換算,會出現同一個值的情況。處理這種情況的一種方法是,拉出一個鍊錶

哈希表這種結構適用於只有等值查詢的場景

有序數組在等值查詢和範圍查詢場景中的效能就都非常優秀

如果只看查詢效率,有序數組就很好。但是,在需要更新資料的時候就麻煩了,你往中間插入一個記錄就必須得挪動後面所有的記錄,成本太高

##有序數組索引只適用於靜態儲存引擎

二元搜尋樹的特徵是:每個節點的左兒子小於父節點,父節點又小於右兒子

當然為了維持O(log(N)) 的查詢複雜度,你就需要保持這棵樹是平衡二元樹。為了做這個 保證,更新的時間複雜度也是 O(log(N))

二元樹是搜尋效率最高的,但是實際上大多數的資料庫儲存卻不使用二元樹。原因是,索引不只存在記憶體中,還要寫到磁碟上

為了讓一個查詢盡量少讀磁碟,就必須讓查詢過程存取盡量少的資料區塊。那麼,我們就不應該使用二元樹,而是要使用「N 叉」樹。這裡,「N 叉」樹中的「N」取決於資料塊的大小

N 叉樹由於在讀寫上的效能優點,以及適配磁碟的存取模式,已經被廣泛應用在資料庫引擎中了

InnoDB 的索引模型

在InnoDB 中,表都是根據主鍵順序以索引的形式存放的,這種儲存方式的表稱為索引組織表。 InnoDB 使用了B 樹索引模型,所以資料都是儲存在B 樹中的

每個索引在InnoDB 裡面對應一棵B 樹

根據葉子節點的內容,索引類型分為主鍵索引和非主鍵索引

主鍵索引的葉子節點存的是整行資料。在 InnoDB 裡,主鍵索引也被稱為叢集索引

非主鍵索引的葉子節點內容是主鍵的值。在 InnoDB 裡,非主鍵索引也被稱為二級索引

基於非主鍵索引的查詢需要多掃描一棵索引樹(回表)。因此,我們在應用程式中應該盡量使用主鍵查詢

#索引維護

B 樹為了維護索引有序性,在插入新值的時候需要做必要的維護

如果新插入的ID 值比原來的小,就相對麻煩了,需要邏輯上挪動後面的數據,空出位置

而更糟的情況是,如果所在的資料頁已經滿了,根據B 樹的演算法,這時候需要申請一個新的資料頁,然後挪動部分資料過去。這個過程稱為頁分裂。在這種情況下,性能自然會受影響。

除了效能外,頁分裂操作也會影響資料頁的使用率。原本放在一個頁的數據,現在分到兩個頁中,整體空間利用率降低約 50%。

當然有分裂就有合併。當相鄰兩頁由於刪除了數據,利用率很低之後,會將數據頁做合 並。合併的過程,可以認為是分裂過程的逆過程

自增主鍵的插入資料模式,正符合了我們前面提到的遞增插入的場景。每次插 入一筆新記錄,都是追加操作,都不涉及到挪動其他記錄,也不會觸發葉子節點的分裂。

而有業務邏輯的欄位做主鍵,則往往不容易保證有序插入,這樣寫資料成本相對較高

主鍵長度越小,普通索引的葉子節點就越小,普通索引佔用的空間也就越小

所以,從效能和儲存空間方面考量,自增主鍵往往是更合理的選擇

#有沒有什麼場景適合用業務欄位直接做主鍵的呢?還是有的。例如,有些業務的場景需求 是這樣的:

1.只有一個索引;

2.該索引必須是唯一索引。

這就是典型的 KV 場景

覆蓋索引

#

如果执行的语句是 select ID from t ,这时只需要查 ID 的 值,而 ID 的值已经在 k 索引树上了,因此可以直接提供查询结果,不需要回表。也就是说,在这个查询里面,索引 k 已经“覆盖了”我们的查询需求,我们称为覆盖索引

由于覆盖索引可以减少树的搜索次数,显著提升查询性能,所以使用覆盖索引是一个常用的性能优化手段

索引下推

满足最左前缀原则的时候,最左前缀可以用于在索引中定位记录。这时,你可能要问,那些不符合最左前缀的部分,会怎么样呢?

MySQL 5.6 引入的索引下推优化, 可以在索引遍历过 程中,对索引中包含的字段先做判断,直接过滤掉不满足条件的记录,减少回表次数

最左前缀原则

不只是索引的全部定义,只要满足最左前缀,就可以利用索引来加速检索

在建立联合索引的时候,如何安排索引内的字段顺序?

这里我们的评估标准是,索引的复用能力。因为可以支持最左前缀,所以当已经有了 (a,b) 这个联合索引后,一般就不需要单独在 a 上建立索引了。因此,第一原则是,如果通过调整顺序,可以少维护一个索引,那么这个顺序往往就是需要优先考虑采用的

前缀索引

利用最左前缀原则可以定义字符串的一部分作为索引。默认地,如果你创建索引的语句不指定前缀长度,那么索引就会包含整个字符串

但,这同时带来的损失是,可能会增加额外的记录扫描次数,因为索引相同需要进一步比较

使用前缀索引,定义好长度,就可以做到既节省空间,又不用额外增加太多的查 询成本

可以通过统计索引上有多少个不同的值来判断要使用多长的前缀,从而减少扫描次数

前缀索引对覆盖索引的影响

使用前缀索引就用不上覆盖索引对查询性能的优化了,这也是你在选择是否使用前缀索引时需要考虑的一个因素

倒序存储和hash存储

对于类似于邮箱这样的字段来说,使用前缀索引的效果可能还不错。但是,遇到前缀的区 分度不够好的情况时,我们要怎么办呢?

第一种方式是使用倒序存储。如果你存储身份证号的时候把它倒过来存

第二种方式是使用 hash 字段。你可以在表上再创建一个整数字段,来保存身份证的校验码,同时在这个字段上创建索引

免费学习视频教程推荐:mysql视频教程

以上是mysql索引詳解(總結)的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述
本文轉載於:oschina。如有侵權,請聯絡admin@php.cn刪除
MySQL和其他SQL方言之間的語法有什麼區別?MySQL和其他SQL方言之間的語法有什麼區別?Apr 27, 2025 am 12:26 AM

mysqldiffersfromothersqldialectsinsyntaxforlimit,自動啟動,弦樂範圍,子征服和表面上分析。 1)MySqluessLipslimit,whilesqlserverusestopopandoraclesrontersrontsrontsrontsronnum.2)

什麼是mysql分區?什麼是mysql分區?Apr 27, 2025 am 12:23 AM

MySQL分區能提升性能和簡化維護。 1)通過按特定標準(如日期範圍)將大表分成小塊,2)物理上將數據分成獨立文件,3)查詢時MySQL可專注於相關分區,4)查詢優化器可跳過不相關分區,5)選擇合適的分區策略並定期維護是關鍵。

您如何在MySQL中授予和撤銷特權?您如何在MySQL中授予和撤銷特權?Apr 27, 2025 am 12:21 AM

在MySQL中,如何授予和撤銷權限? 1.使用GRANT語句授予權限,如GRANTALLPRIVILEGESONdatabase_name.TO'username'@'host';2.使用REVOKE語句撤銷權限,如REVOKEALLPRIVILEGESONdatabase_name.FROM'username'@'host',確保及時溝通權限變更。

說明InnoDB和Myisam存儲引擎之間的差異。說明InnoDB和Myisam存儲引擎之間的差異。Apr 27, 2025 am 12:20 AM

InnoDB適合需要事務支持和高並發性的應用,MyISAM適合讀多寫少的應用。 1.InnoDB支持事務和行級鎖,適用於電商和銀行系統。 2.MyISAM提供快速讀取和索引,適合博客和內容管理系統。

MySQL中有哪些不同類型的連接?MySQL中有哪些不同類型的連接?Apr 27, 2025 am 12:13 AM

MySQL中有四種主要的JOIN類型:INNERJOIN、LEFTJOIN、RIGHTJOIN和FULLOUTERJOIN。 1.INNERJOIN返回兩個表中符合JOIN條件的所有行。 2.LEFTJOIN返回左表中的所有行,即使右表中沒有匹配的行。 3.RIGHTJOIN與LEFTJOIN相反,返回右表中的所有行。 4.FULLOUTERJOIN返回兩個表中所有符合或不符合JOIN條件的行。

MySQL中有哪些不同的存儲引擎?MySQL中有哪些不同的存儲引擎?Apr 26, 2025 am 12:27 AM

mysqloffersvariousStorageengines,每個suitedfordferentusecases:1)InnodBisidealForapplicationsNeedingingAcidComplianCeanDhighConcurncurnency,supportingtransactionsancions and foreignkeys.2)myisamisbestforread-Heavy-Heavywyworks,lackingtransactionsactionsacupport.3)記憶

MySQL中有哪些常見的安全漏洞?MySQL中有哪些常見的安全漏洞?Apr 26, 2025 am 12:27 AM

MySQL中常見的安全漏洞包括SQL注入、弱密碼、權限配置不當和未更新的軟件。 1.SQL注入可以通過使用預處理語句防止。 2.弱密碼可以通過強制使用強密碼策略避免。 3.權限配置不當可以通過定期審查和調整用戶權限解決。 4.未更新的軟件可以通過定期檢查和更新MySQL版本來修補。

您如何確定MySQL中的慢速查詢?您如何確定MySQL中的慢速查詢?Apr 26, 2025 am 12:15 AM

在MySQL中識別慢查詢可以通過啟用慢查詢日誌並設置閾值來實現。 1.啟用慢查詢日誌並設置閾值。 2.查看和分析慢查詢日誌文件,使用工具如mysqldumpslow或pt-query-digest進行深入分析。 3.優化慢查詢可以通過索引優化、查詢重寫和避免使用SELECT*來實現。

See all articles

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

Dreamweaver Mac版

Dreamweaver Mac版

視覺化網頁開發工具

SecLists

SecLists

SecLists是最終安全測試人員的伙伴。它是一個包含各種類型清單的集合,這些清單在安全評估過程中經常使用,而且都在一個地方。 SecLists透過方便地提供安全測試人員可能需要的所有列表,幫助提高安全測試的效率和生產力。清單類型包括使用者名稱、密碼、URL、模糊測試有效載荷、敏感資料模式、Web shell等等。測試人員只需將此儲存庫拉到新的測試機上,他就可以存取所需的每種類型的清單。

mPDF

mPDF

mPDF是一個PHP庫,可以從UTF-8編碼的HTML產生PDF檔案。原作者Ian Back編寫mPDF以從他的網站上「即時」輸出PDF文件,並處理不同的語言。與原始腳本如HTML2FPDF相比,它的速度較慢,並且在使用Unicode字體時產生的檔案較大,但支援CSS樣式等,並進行了大量增強。支援幾乎所有語言,包括RTL(阿拉伯語和希伯來語)和CJK(中日韓)。支援嵌套的區塊級元素(如P、DIV),

SublimeText3 英文版

SublimeText3 英文版

推薦:為Win版本,支援程式碼提示!

VSCode Windows 64位元 下載

VSCode Windows 64位元 下載

微軟推出的免費、功能強大的一款IDE編輯器