這篇文章主要介紹“MySQL的索引結(jié)構(gòu)為什么使用B+樹”,在日常操作中,相信很多人在MySQL的索引結(jié)構(gòu)為什么使用B+樹問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”MySQL的索引結(jié)構(gòu)為什么使用B+樹”的疑惑有所幫助!接下來,請跟著小編一起來學(xué)習(xí)吧!
公司主營業(yè)務(wù):成都網(wǎng)站建設(shè)、成都網(wǎng)站設(shè)計(jì)、移動網(wǎng)站開發(fā)等業(yè)務(wù)。幫助企業(yè)客戶真正實(shí)現(xiàn)互聯(lián)網(wǎng)宣傳,提高企業(yè)的競爭能力。創(chuàng)新互聯(lián)建站是一支青春激揚(yáng)、勤奮敬業(yè)、活力青春激揚(yáng)、勤奮敬業(yè)、活力澎湃、和諧高效的團(tuán)隊(duì)。公司秉承以“開放、自由、嚴(yán)謹(jǐn)、自律”為核心的企業(yè)文化,感謝他們對我們的高要求,感謝他們從不同領(lǐng)域給我們帶來的挑戰(zhàn),讓我們激情的團(tuán)隊(duì)有機(jī)會用頭腦與智慧不斷的給客戶帶來驚喜。創(chuàng)新互聯(lián)建站推出襄城免費(fèi)做網(wǎng)站回饋大家。
在MySQL中,無論是Innodb還是MyIsam,都使用了B+樹作索引結(jié)構(gòu)(這里不考慮hash等其他索引)。本文將從最普通的二叉查找樹開始,逐步說明各種樹解決的問題以及面臨的新問題,從而說明MySQL為什么選擇B+樹作為索引結(jié)構(gòu)。
一、二叉查找樹(BST):不平衡
二、平衡二叉樹(AVL):旋轉(zhuǎn)耗時(shí)
三、紅黑樹:樹太高
四、B樹:為磁盤而生
五、B+樹
六、感受B+樹的威力
七、總結(jié)
二叉查找樹(BST,Binary Search Tree),也叫二叉排序樹,在二叉樹的基礎(chǔ)上需要滿足:任意節(jié)點(diǎn)的左子樹上所有節(jié)點(diǎn)值不大于根節(jié)點(diǎn)的值,任意節(jié)點(diǎn)的右子樹上所有節(jié)點(diǎn)值不小于根節(jié)點(diǎn)的值。如下是一顆BST(圖片來源)。
當(dāng)需要快速查找時(shí),將數(shù)據(jù)存儲在BST是一種常見的選擇,因?yàn)榇藭r(shí)查詢時(shí)間取決于樹高,平均時(shí)間復(fù)雜度是O(lgn)。然而BST可能長歪而變得不平衡,如下圖所示(圖片來源),此時(shí)BST退化為鏈表,時(shí)間復(fù)雜度退化為O(n)。
為了解決這個(gè)問題,引入了平衡二叉樹。
AVL樹是嚴(yán)格的平衡二叉樹,所有節(jié)點(diǎn)的左右子樹高度差不能超過1;AVL樹查找、插入和刪除在平均和最壞情況下都是O(lgn)。
AVL實(shí)現(xiàn)平衡的關(guān)鍵在于旋轉(zhuǎn)操作:插入和刪除可能破壞二叉樹的平衡,此時(shí)需要通過一次或多次樹旋轉(zhuǎn)來重新平衡這個(gè)樹。當(dāng)插入數(shù)據(jù)時(shí),最多只需要1次旋轉(zhuǎn)(單旋轉(zhuǎn)或雙旋轉(zhuǎn));但是當(dāng)刪除數(shù)據(jù)時(shí),會導(dǎo)致樹失衡,AVL需要維護(hù)從被刪除節(jié)點(diǎn)到根節(jié)點(diǎn)這條路徑上所有節(jié)點(diǎn)的平衡,旋轉(zhuǎn)的量級為O(lgn)。
由于旋轉(zhuǎn)的耗時(shí),AVL樹在刪除數(shù)據(jù)時(shí)效率很低;在刪除操作較多時(shí),維護(hù)平衡所需的代價(jià)可能高于其帶來的好處,因此AVL實(shí)際使用并不廣泛。
與AVL樹相比,紅黑樹并不追求嚴(yán)格的平衡,而是大致的平衡:只是確保從根到葉子的最長的可能路徑不多于最短的可能路徑的兩倍長。從實(shí)現(xiàn)來看,紅黑樹最大的特點(diǎn)是每個(gè)節(jié)點(diǎn)都屬于兩種顏色(紅色或黑色)之一,且節(jié)點(diǎn)顏色的劃分需要滿足特定的規(guī)則(具體規(guī)則略)。紅黑樹示例如下(圖片來源):
與AVL樹相比,紅黑樹的查詢效率會有所下降,這是因?yàn)闃涞钠胶庑宰儾?,高度更高。但紅黑樹的刪除效率大大提高了,因?yàn)榧t黑樹同時(shí)引入了顏色,當(dāng)插入或刪除數(shù)據(jù)時(shí),只需要進(jìn)行O(1)次數(shù)的旋轉(zhuǎn)以及變色就能保證基本的平衡,不需要像AVL樹進(jìn)行O(lgn)次數(shù)的旋轉(zhuǎn)??偟膩碚f,紅黑樹的統(tǒng)計(jì)性能高于AVL。
因此,在實(shí)際應(yīng)用中,AVL樹的使用相對較少,而紅黑樹的使用非常廣泛。例如,Java中的TreeMap使用紅黑樹存儲排序鍵值對;Java8中的HashMap使用鏈表+紅黑樹解決哈希沖突問題(當(dāng)沖突節(jié)點(diǎn)較少時(shí),使用鏈表,當(dāng)沖突節(jié)點(diǎn)較多時(shí),使用紅黑樹)。
對于數(shù)據(jù)在內(nèi)存中的情況(如上述的TreeMap和HashMap),紅黑樹的表現(xiàn)是非常優(yōu)異的。但是對于數(shù)據(jù)在磁盤等輔助存儲設(shè)備中的情況(如MySQL等數(shù)據(jù)庫),紅黑樹并不擅長,因?yàn)榧t黑樹長得還是太高了。當(dāng)數(shù)據(jù)在磁盤中時(shí),磁盤IO會成為最大的性能瓶頸,設(shè)計(jì)的目標(biāo)應(yīng)該是盡量減少IO次數(shù);而樹的高度越高,增刪改查所需要的IO次數(shù)也越多,會嚴(yán)重影響性能。
B樹也稱B-樹(其中-不是減號),是為磁盤等輔存設(shè)備設(shè)計(jì)的多路平衡查找樹,與二叉樹相比,B樹的每個(gè)非葉節(jié)點(diǎn)可以有多個(gè)子樹。因此,當(dāng)總節(jié)點(diǎn)數(shù)量相同時(shí),B樹的高度遠(yuǎn)遠(yuǎn)小于AVL樹和紅黑樹(B樹是一顆“矮胖子”),磁盤IO次數(shù)大大減少。
定義B樹最重要的概念是階數(shù)(Order),對于一顆m階B樹,需要滿足以下條件:
可以看出,B樹的定義,主要是對非葉結(jié)點(diǎn)的子節(jié)點(diǎn)數(shù)量和記錄數(shù)量的限制。
下圖是一個(gè)3階B樹的例子(圖片來源):
B樹的優(yōu)勢除了樹高小,還有對訪問局部性原理的利用。所謂局部性原理,是指當(dāng)一個(gè)數(shù)據(jù)被使用時(shí),其附近的數(shù)據(jù)有較大概率在短時(shí)間內(nèi)被使用。B樹將鍵相近的數(shù)據(jù)存儲在同一個(gè)節(jié)點(diǎn),當(dāng)訪問其中某個(gè)數(shù)據(jù)時(shí),數(shù)據(jù)庫會將該整個(gè)節(jié)點(diǎn)讀到緩存中;當(dāng)它臨近的數(shù)據(jù)緊接著被訪問時(shí),可以直接在緩存中讀取,無需進(jìn)行磁盤IO;換句話說,B樹的緩存命中率更高。
B樹在數(shù)據(jù)庫中有一些應(yīng)用,如MongoDB的索引使用了B樹結(jié)構(gòu)。但是在很多數(shù)據(jù)庫應(yīng)用中,使用了是B樹的變種B+樹。
B+樹也是多路平衡查找樹,其與B樹的區(qū)別主要在于:
由此,B+樹與B樹相比,有以下優(yōu)勢:
B+樹也存在劣勢:由于鍵會重復(fù)出現(xiàn),因此會占用更多的空間。但是與帶來的性能優(yōu)勢相比,空間劣勢往往可以接受,因此B+樹的在數(shù)據(jù)庫中的使用比B樹更加廣泛。
前面說到,B樹/B+樹與紅黑樹等二叉樹相比,最大的優(yōu)勢在于樹高更小。實(shí)際上,對于Innodb的B+索引來說,樹的高度一般在2-4層。下面來進(jìn)行一些具體的估算。
樹的高度是由階數(shù)決定的,階數(shù)越大樹越矮;而階數(shù)的大小又取決于每個(gè)節(jié)點(diǎn)可以存儲多少條記錄。Innodb中每個(gè)節(jié)點(diǎn)使用一個(gè)頁(page),頁的大小為16KB,其中元數(shù)據(jù)只占大約128字節(jié)左右(包括文件管理頭信息、頁面頭信息等等),大多數(shù)空間都用來存儲數(shù)據(jù)。
對于一顆3層B+樹,第一層(根節(jié)點(diǎn))有1個(gè)頁面,可以存儲1000條記錄;第二層有1000個(gè)頁面,可以存儲10001000條記錄;第三層(葉節(jié)點(diǎn))有10001000個(gè)頁面,每個(gè)頁面可以存儲100條記錄,因此可以存儲10001000100條記錄,即1億條。而對于二叉樹,存儲1億條記錄則需要26層左右。
最后,總結(jié)一下各種樹解決的問題以及面臨的新問題:
二叉查找樹(BST):解決了排序的基本問題,但是由于無法保證平衡,可能退化為鏈表;
平衡二叉樹(AVL):通過旋轉(zhuǎn)解決了平衡的問題,但是旋轉(zhuǎn)操作效率太低;
紅黑樹:通過舍棄嚴(yán)格的平衡和引入紅黑節(jié)點(diǎn),解決了AVL旋轉(zhuǎn)效率過低的問題,但是在磁盤等場景下,樹仍然太高,IO次數(shù)太多;
B樹:通過將二叉樹改為多路平衡查找樹,解決了樹過高的問題;
B+樹:在B樹的基礎(chǔ)上,將非葉節(jié)點(diǎn)改造為不存儲數(shù)據(jù)的純索引節(jié)點(diǎn),進(jìn)一步降低了樹的高度;此外將葉節(jié)點(diǎn)使用指針連接成鏈表,范圍查詢更加高效。
到此,關(guān)于“MySQL的索引結(jié)構(gòu)為什么使用B+樹”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識,請繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬?shí)用的文章!
網(wǎng)站題目:MySQL的索引結(jié)構(gòu)為什么使用B+樹
文章網(wǎng)址:http://jinyejixie.com/article8/jjioip.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站設(shè)計(jì)公司、、網(wǎng)站制作、網(wǎng)站營銷、企業(yè)網(wǎng)站制作、網(wǎng)站收錄
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)