日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 运维知识 > 数据库 >内容正文

数据库

MySQL索引背后的数据结构及算法原理--转

發(fā)布時間:2025/4/5 数据库 28 豆豆
生活随笔 收集整理的這篇文章主要介紹了 MySQL索引背后的数据结构及算法原理--转 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.
MySQL索引背后的數(shù)據(jù)結(jié)構(gòu)及算法原理 作者 張洋 | 發(fā)布于 2011-10-18 MySQL?索引?B樹?優(yōu)化 原文地址:http://blog.codinglabs.org/articles/theory-of-mysql-index.html

摘要

本文以MySQL數(shù)據(jù)庫為研究對象,討論與數(shù)據(jù)庫索引相關(guān)的一些話題。特別需要說明的是,MySQL支持諸多存儲引擎,而各種存儲引擎對索引的支持也各不相同,因此MySQL數(shù)據(jù)庫支持多種索引類型,如BTree索引,哈希索引,全文索引等等。為了避免混亂,本文將只關(guān)注于BTree索引,因為這是平常使用MySQL時主要打交道的索引,至于哈希索引和全文索引本文暫不討論。

文章主要內(nèi)容分為三個部分。

第一部分主要從數(shù)據(jù)結(jié)構(gòu)及算法理論層面討論MySQL數(shù)據(jù)庫索引的數(shù)理基礎(chǔ)。

第二部分結(jié)合MySQL數(shù)據(jù)庫中MyISAM和InnoDB數(shù)據(jù)存儲引擎中索引的架構(gòu)實現(xiàn)討論聚集索引、非聚集索引及覆蓋索引等話題。

第三部分根據(jù)上面的理論基礎(chǔ),討論MySQL中高性能使用索引的策略。

數(shù)據(jù)結(jié)構(gòu)及算法基礎(chǔ)

索引的本質(zhì)

MySQL官方對索引的定義為:索引(Index)是幫助MySQL高效獲取數(shù)據(jù)的數(shù)據(jù)結(jié)構(gòu)。提取句子主干,就可以得到索引的本質(zhì):索引是數(shù)據(jù)結(jié)構(gòu)。

我們知道,數(shù)據(jù)庫查詢是數(shù)據(jù)庫的最主要功能之一。我們都希望查詢數(shù)據(jù)的速度能盡可能的快,因此數(shù)據(jù)庫系統(tǒng)的設(shè)計者會從查詢算法的角度進行優(yōu)化。最基本的查詢算法當(dāng)然是順序查找(linear search),這種復(fù)雜度為O(n)的算法在數(shù)據(jù)量很大時顯然是糟糕的,好在計算機科學(xué)的發(fā)展提供了很多更優(yōu)秀的查找算法,例如二分查找(binary search)、二叉樹查找(binary tree search)等。如果稍微分析一下會發(fā)現(xiàn),每種查找算法都只能應(yīng)用于特定的數(shù)據(jù)結(jié)構(gòu)之上,例如二分查找要求被檢索數(shù)據(jù)有序,而二叉樹查找只能應(yīng)用于二叉查找樹上,但是數(shù)據(jù)本身的組織結(jié)構(gòu)不可能完全滿足各種數(shù)據(jù)結(jié)構(gòu)(例如,理論上不可能同時將兩列都按順序進行組織),所以,在數(shù)據(jù)之外,數(shù)據(jù)庫系統(tǒng)還維護著滿足特定查找算法的數(shù)據(jù)結(jié)構(gòu),這些數(shù)據(jù)結(jié)構(gòu)以某種方式引用(指向)數(shù)據(jù),這樣就可以在這些數(shù)據(jù)結(jié)構(gòu)上實現(xiàn)高級查找算法。這種數(shù)據(jù)結(jié)構(gòu),就是索引。

看一個例子:

圖1

圖1展示了一種可能的索引方式。左邊是數(shù)據(jù)表,一共有兩列七條記錄,最左邊的是數(shù)據(jù)記錄的物理地址(注意邏輯上相鄰的記錄在磁盤上也并不是一定物理相鄰的)。為了加快Col2的查找,可以維護一個右邊所示的二叉查找樹,每個節(jié)點分別包含索引鍵值和一個指向?qū)?yīng)數(shù)據(jù)記錄物理地址的指針,這樣就可以運用二叉查找在O(log2n)的復(fù)雜度內(nèi)獲取到相應(yīng)數(shù)據(jù)。

雖然這是一個貨真價實的索引,但是實際的數(shù)據(jù)庫系統(tǒng)幾乎沒有使用二叉查找樹或其進化品種紅黑樹(red-black tree)實現(xiàn)的,原因會在下文介紹。

B-Tree和B+Tree

目前大部分數(shù)據(jù)庫系統(tǒng)及文件系統(tǒng)都采用B-Tree或其變種B+Tree作為索引結(jié)構(gòu),在本文的下一節(jié)會結(jié)合存儲器原理及計算機存取原理討論為什么B-Tree和B+Tree在被如此廣泛用于索引,這一節(jié)先單純從數(shù)據(jù)結(jié)構(gòu)角度描述它們。

B-Tree

為了描述B-Tree,首先定義一條數(shù)據(jù)記錄為一個二元組[key, data],key為記錄的鍵值,對于不同數(shù)據(jù)記錄,key是互不相同的;data為數(shù)據(jù)記錄除key外的數(shù)據(jù)。那么B-Tree是滿足下列條件的數(shù)據(jù)結(jié)構(gòu):

d為大于1的一個正整數(shù),稱為B-Tree的度。

h為一個正整數(shù),稱為B-Tree的高度。

每個非葉子節(jié)點由n-1個key和n個指針組成,其中d<=n<=2d。

每個葉子節(jié)點最少包含一個key和兩個指針,最多包含2d-1個key和2d個指針,葉節(jié)點的指針均為null 。

所有葉節(jié)點具有相同的深度,等于樹高h。

key和指針互相間隔,節(jié)點兩端是指針。

一個節(jié)點中的key從左到右非遞減排列。

所有節(jié)點組成樹結(jié)構(gòu)。

每個指針要么為null,要么指向另外一個節(jié)點。

如果某個指針在節(jié)點node最左邊且不為null,則其指向節(jié)點的所有key小于v(key1),其中v(key1)為node的第一個key的值。

如果某個指針在節(jié)點node最右邊且不為null,則其指向節(jié)點的所有key大于v(keym),其中v(keym)為node的最后一個key的值。

如果某個指針在節(jié)點node的左右相鄰key分別是keyikeyi+1且不為null,則其指向節(jié)點的所有key小于v(keyi+1)且大于v(keyi)。

圖2是一個d=2的B-Tree示意圖。

圖2

由于B-Tree的特性,在B-Tree中按key檢索數(shù)據(jù)的算法非常直觀:首先從根節(jié)點進行二分查找,如果找到則返回對應(yīng)節(jié)點的data,否則對相應(yīng)區(qū)間的指針指向的節(jié)點遞歸進行查找,直到找到節(jié)點或找到null指針,前者查找成功,后者查找失敗。B-Tree上查找算法的偽代碼如下:

  • BTree_Search(node, key){
  • if(node ==null)returnnull;
  • foreach(node.key)
  • {
  • if(node.key[i]== key)return node.data[i];
  • if(node.key[i]> key)returnBTree_Search(point[i]->node);
  • }
  • returnBTree_Search(point[i+1]->node);
  • }
  • data =BTree_Search(root, my_key);
  • 關(guān)于B-Tree有一系列有趣的性質(zhì),例如一個度為d的B-Tree,設(shè)其索引N個key,則其樹高h的上限為logd((N+1)/2),檢索一個key,其查找節(jié)點個數(shù)的漸進復(fù)雜度為O(logdN)。從這點可以看出,B-Tree是一個非常有效率的索引數(shù)據(jù)結(jié)構(gòu)。

    ?

    另外,由于插入刪除新的數(shù)據(jù)記錄會破壞B-Tree的性質(zhì),因此在插入刪除時,需要對樹進行一個分裂、合并、轉(zhuǎn)移等操作以保持B-Tree性質(zhì),本文不打算完整討論B-Tree這些內(nèi)容,因為已經(jīng)有許多資料詳細說明了B-Tree的數(shù)學(xué)性質(zhì)及插入刪除算法,有興趣的朋友可以在本文末的參考文獻一欄找到相應(yīng)的資料進行閱讀。

    B+Tree

    B-Tree有許多變種,其中最常見的是B+Tree,例如MySQL就普遍使用B+Tree實現(xiàn)其索引結(jié)構(gòu)。

    與B-Tree相比,B+Tree有以下不同點:

    每個節(jié)點的指針上限為2d而不是2d+1。

    內(nèi)節(jié)點不存儲data,只存儲key;葉子節(jié)點不存儲指針。

    圖3是一個簡單的B+Tree示意。

    圖3

    由于并不是所有節(jié)點都具有相同的域,因此B+Tree中葉節(jié)點和內(nèi)節(jié)點一般大小不同。這點與B-Tree不同,雖然B-Tree中不同節(jié)點存放的key和指針可能數(shù)量不一致,但是每個節(jié)點的域和上限是一致的,所以在實現(xiàn)中B-Tree往往對每個節(jié)點申請同等大小的空間。

    一般來說,B+Tree比B-Tree更適合實現(xiàn)外存儲索引結(jié)構(gòu),具體原因與外存儲器原理及計算機存取原理有關(guān),將在下面討論。

    帶有順序訪問指針的B+Tree

    一般在數(shù)據(jù)庫系統(tǒng)或文件系統(tǒng)中使用的B+Tree結(jié)構(gòu)都在經(jīng)典B+Tree的基礎(chǔ)上進行了優(yōu)化,增加了順序訪問指針。

    圖4

    如圖4所示,在B+Tree的每個葉子節(jié)點增加一個指向相鄰葉子節(jié)點的指針,就形成了帶有順序訪問指針的B+Tree。做這個優(yōu)化的目的是為了提高區(qū)間訪問的性能,例如圖4中如果要查詢key為從18到49的所有數(shù)據(jù)記錄,當(dāng)找到18后,只需順著節(jié)點和指針順序遍歷就可以一次性訪問到所有數(shù)據(jù)節(jié)點,極大提到了區(qū)間查詢效率。

    這一節(jié)對B-Tree和B+Tree進行了一個簡單的介紹,下一節(jié)結(jié)合存儲器存取原理介紹為什么目前B+Tree是數(shù)據(jù)庫系統(tǒng)實現(xiàn)索引的首選數(shù)據(jù)結(jié)構(gòu)。

    為什么使用B-Tree(B+Tree)

    上文說過,紅黑樹等數(shù)據(jù)結(jié)構(gòu)也可以用來實現(xiàn)索引,但是文件系統(tǒng)及數(shù)據(jù)庫系統(tǒng)普遍采用B-/+Tree作為索引結(jié)構(gòu),這一節(jié)將結(jié)合計算機組成原理相關(guān)知識討論B-/+Tree作為索引的理論基礎(chǔ)。

    一般來說,索引本身也很大,不可能全部存儲在內(nèi)存中,因此索引往往以索引文件的形式存儲的磁盤上。這樣的話,索引查找過程中就要產(chǎn)生磁盤I/O消耗,相對于內(nèi)存存取,I/O存取的消耗要高幾個數(shù)量級,所以評價一個數(shù)據(jù)結(jié)構(gòu)作為索引的優(yōu)劣最重要的指標(biāo)就是在查找過程中磁盤I/O操作次數(shù)的漸進復(fù)雜度。換句話說,索引的結(jié)構(gòu)組織要盡量減少查找過程中磁盤I/O的存取次數(shù)。下面先介紹內(nèi)存和磁盤存取原理,然后再結(jié)合這些原理分析B-/+Tree作為索引的效率。

    主存存取原理

    目前計算機使用的主存基本都是隨機讀寫存儲器(RAM),現(xiàn)代RAM的結(jié)構(gòu)和存取原理比較復(fù)雜,這里本文拋卻具體差別,抽象出一個十分簡單的存取模型來說明RAM的工作原理。

    圖5

    從抽象角度看,主存是一系列的存儲單元組成的矩陣,每個存儲單元存儲固定大小的數(shù)據(jù)。每個存儲單元有唯一的地址,現(xiàn)代主存的編址規(guī)則比較復(fù)雜,這里將其簡化成一個二維地址:通過一個行地址和一個列地址可以唯一定位到一個存儲單元。圖5展示了一個4 x 4的主存模型。

    主存的存取過程如下:

    當(dāng)系統(tǒng)需要讀取主存時,則將地址信號放到地址總線上傳給主存,主存讀到地址信號后,解析信號并定位到指定存儲單元,然后將此存儲單元數(shù)據(jù)放到數(shù)據(jù)總線上,供其它部件讀取。

    寫主存的過程類似,系統(tǒng)將要寫入單元地址和數(shù)據(jù)分別放在地址總線和數(shù)據(jù)總線上,主存讀取兩個總線的內(nèi)容,做相應(yīng)的寫操作。

    這里可以看出,主存存取的時間僅與存取次數(shù)呈線性關(guān)系,因為不存在機械操作,兩次存取的數(shù)據(jù)的“距離”不會對時間有任何影響,例如,先取A0再取A1和先取A0再取D3的時間消耗是一樣的。

    磁盤存取原理

    上文說過,索引一般以文件形式存儲在磁盤上,索引檢索需要磁盤I/O操作。與主存不同,磁盤I/O存在機械運動耗費,因此磁盤I/O的時間消耗是巨大的。

    圖6是磁盤的整體結(jié)構(gòu)示意圖。

    圖6

    一個磁盤由大小相同且同軸的圓形盤片組成,磁盤可以轉(zhuǎn)動(各個磁盤必須同步轉(zhuǎn)動)。在磁盤的一側(cè)有磁頭支架,磁頭支架固定了一組磁頭,每個磁頭負責(zé)存取一個磁盤的內(nèi)容。磁頭不能轉(zhuǎn)動,但是可以沿磁盤半徑方向運動(實際是斜切向運動),每個磁頭同一時刻也必須是同軸的,即從正上方向下看,所有磁頭任何時候都是重疊的(不過目前已經(jīng)有多磁頭獨立技術(shù),可不受此限制)。

    圖7是磁盤結(jié)構(gòu)的示意圖。

    圖7

    盤片被劃分成一系列同心環(huán),圓心是盤片中心,每個同心環(huán)叫做一個磁道,所有半徑相同的磁道組成一個柱面。磁道被沿半徑線劃分成一個個小的段,每個段叫做一個扇區(qū),每個扇區(qū)是磁盤的最小存儲單元。為了簡單起見,我們下面假設(shè)磁盤只有一個盤片和一個磁頭。

    當(dāng)需要從磁盤讀取數(shù)據(jù)時,系統(tǒng)會將數(shù)據(jù)邏輯地址傳給磁盤,磁盤的控制電路按照尋址邏輯將邏輯地址翻譯成物理地址,即確定要讀的數(shù)據(jù)在哪個磁道,哪個扇區(qū)。為了讀取這個扇區(qū)的數(shù)據(jù),需要將磁頭放到這個扇區(qū)上方,為了實現(xiàn)這一點,磁頭需要移動對準(zhǔn)相應(yīng)磁道,這個過程叫做尋道,所耗費時間叫做尋道時間,然后磁盤旋轉(zhuǎn)將目標(biāo)扇區(qū)旋轉(zhuǎn)到磁頭下,這個過程耗費的時間叫做旋轉(zhuǎn)時間。

    局部性原理與磁盤預(yù)讀

    由于存儲介質(zhì)的特性,磁盤本身存取就比主存慢很多,再加上機械運動耗費,磁盤的存取速度往往是主存的幾百分分之一,因此為了提高效率,要盡量減少磁盤I/O。為了達到這個目的,磁盤往往不是嚴格按需讀取,而是每次都會預(yù)讀,即使只需要一個字節(jié),磁盤也會從這個位置開始,順序向后讀取一定長度的數(shù)據(jù)放入內(nèi)存。這樣做的理論依據(jù)是計算機科學(xué)中著名的局部性原理:

    當(dāng)一個數(shù)據(jù)被用到時,其附近的數(shù)據(jù)也通常會馬上被使用。

    程序運行期間所需要的數(shù)據(jù)通常比較集中。

    由于磁盤順序讀取的效率很高(不需要尋道時間,只需很少的旋轉(zhuǎn)時間),因此對于具有局部性的程序來說,預(yù)讀可以提高I/O效率。

    預(yù)讀的長度一般為頁(page)的整倍數(shù)。頁是計算機管理存儲器的邏輯塊,硬件及操作系統(tǒng)往往將主存和磁盤存儲區(qū)分割為連續(xù)的大小相等的塊,每個存儲塊稱為一頁(在許多操作系統(tǒng)中,頁得大小通常為4k),主存和磁盤以頁為單位交換數(shù)據(jù)。當(dāng)程序要讀取的數(shù)據(jù)不在主存中時,會觸發(fā)一個缺頁異常,此時系統(tǒng)會向磁盤發(fā)出讀盤信號,磁盤會找到數(shù)據(jù)的起始位置并向后連續(xù)讀取一頁或幾頁載入內(nèi)存中,然后異常返回,程序繼續(xù)運行。

    B-/+Tree索引的性能分析

    到這里終于可以分析B-/+Tree索引的性能了。

    上文說過一般使用磁盤I/O次數(shù)評價索引結(jié)構(gòu)的優(yōu)劣。先從B-Tree分析,根據(jù)B-Tree的定義,可知檢索一次最多需要訪問h個節(jié)點。數(shù)據(jù)庫系統(tǒng)的設(shè)計者巧妙利用了磁盤預(yù)讀原理,將一個節(jié)點的大小設(shè)為等于一個頁,這樣每個節(jié)點只需要一次I/O就可以完全載入。為了達到這個目的,在實際實現(xiàn)B-Tree還需要使用如下技巧:

    每次新建節(jié)點時,直接申請一個頁的空間,這樣就保證一個節(jié)點物理上也存儲在一個頁里,加之計算機存儲分配都是按頁對齊的,就實現(xiàn)了一個node只需一次I/O。

    B-Tree中一次檢索最多需要h-1次I/O(根節(jié)點常駐內(nèi)存),漸進復(fù)雜度為O(h)=O(logdN)。一般實際應(yīng)用中,出度d是非常大的數(shù)字,通常超過100,因此h非常小(通常不超過3)。

    綜上所述,用B-Tree作為索引結(jié)構(gòu)效率是非常高的。

    而紅黑樹這種結(jié)構(gòu),h明顯要深的多。由于邏輯上很近的節(jié)點(父子)物理上可能很遠,無法利用局部性,所以紅黑樹的I/O漸進復(fù)雜度也為O(h),效率明顯比B-Tree差很多。

    上文還說過,B+Tree更適合外存索引,原因和內(nèi)節(jié)點出度d有關(guān)。從上面分析可以看到,d越大索引的性能越好,而出度的上限取決于節(jié)點內(nèi)key和data的大小:

    dmax=floor(pagesize/(keysize+datasize+pointsize))

    floor表示向下取整。由于B+Tree內(nèi)節(jié)點去掉了data域,因此可以擁有更大的出度,擁有更好的性能。

    這一章從理論角度討論了與索引相關(guān)的數(shù)據(jù)結(jié)構(gòu)與算法問題,下一章將討論B+Tree是如何具體實現(xiàn)為MySQL中索引,同時將結(jié)合MyISAM和InnDB存儲引擎介紹非聚集索引和聚集索引兩種不同的索引實現(xiàn)形式。

    MySQL索引實現(xiàn)

    在MySQL中,索引屬于存儲引擎級別的概念,不同存儲引擎對索引的實現(xiàn)方式是不同的,本文主要討論MyISAM和InnoDB兩個存儲引擎的索引實現(xiàn)方式。

    MyISAM索引實現(xiàn)

    MyISAM引擎使用B+Tree作為索引結(jié)構(gòu),葉節(jié)點的data域存放的是數(shù)據(jù)記錄的地址。下圖是MyISAM索引的原理圖:

    圖8

    這里設(shè)表一共有三列,假設(shè)我們以Col1為主鍵,則圖8是一個MyISAM表的主索引(Primary key)示意??梢钥闯鯩yISAM的索引文件僅僅保存數(shù)據(jù)記錄的地址。在MyISAM中,主索引和輔助索引(Secondary key)在結(jié)構(gòu)上沒有任何區(qū)別,只是主索引要求key是唯一的,而輔助索引的key可以重復(fù)。如果我們在Col2上建立一個輔助索引,則此索引的結(jié)構(gòu)如下圖所示:

    圖9

    同樣也是一顆B+Tree,data域保存數(shù)據(jù)記錄的地址。因此,MyISAM中索引檢索的算法為首先按照B+Tree搜索算法搜索索引,如果指定的Key存在,則取出其data域的值,然后以data域的值為地址,讀取相應(yīng)數(shù)據(jù)記錄。

    MyISAM的索引方式也叫做“非聚集”的,之所以這么稱呼是為了與InnoDB的聚集索引區(qū)分。

    InnoDB索引實現(xiàn)

    雖然InnoDB也使用B+Tree作為索引結(jié)構(gòu),但具體實現(xiàn)方式卻與MyISAM截然不同。

    第一個重大區(qū)別是InnoDB的數(shù)據(jù)文件本身就是索引文件。從上文知道,MyISAM索引文件和數(shù)據(jù)文件是分離的,索引文件僅保存數(shù)據(jù)記錄的地址。而在InnoDB中,表數(shù)據(jù)文件本身就是按B+Tree組織的一個索引結(jié)構(gòu),這棵樹的葉節(jié)點data域保存了完整的數(shù)據(jù)記錄。這個索引的key是數(shù)據(jù)表的主鍵,因此InnoDB表數(shù)據(jù)文件本身就是主索引。

    圖10

    圖10是InnoDB主索引(同時也是數(shù)據(jù)文件)的示意圖,可以看到葉節(jié)點包含了完整的數(shù)據(jù)記錄。這種索引叫做聚集索引。因為InnoDB的數(shù)據(jù)文件本身要按主鍵聚集,所以InnoDB要求表必須有主鍵(MyISAM可以沒有),如果沒有顯式指定,則MySQL系統(tǒng)會自動選擇一個可以唯一標(biāo)識數(shù)據(jù)記錄的列作為主鍵,如果不存在這種列,則MySQL自動為InnoDB表生成一個隱含字段作為主鍵,這個字段長度為6個字節(jié),類型為長整形。

    第二個與MyISAM索引的不同是InnoDB的輔助索引data域存儲相應(yīng)記錄主鍵的值而不是地址。換句話說,InnoDB的所有輔助索引都引用主鍵作為data域。例如,圖11為定義在Col3上的一個輔助索引:

    圖11

    這里以英文字符的ASCII碼作為比較準(zhǔn)則。聚集索引這種實現(xiàn)方式使得按主鍵的搜索十分高效,但是輔助索引搜索需要檢索兩遍索引:首先檢索輔助索引獲得主鍵,然后用主鍵到主索引中檢索獲得記錄。

    了解不同存儲引擎的索引實現(xiàn)方式對于正確使用和優(yōu)化索引都非常有幫助,例如知道了InnoDB的索引實現(xiàn)后,就很容易明白為什么不建議使用過長的字段作為主鍵,因為所有輔助索引都引用主索引,過長的主索引會令輔助索引變得過大。再例如,用非單調(diào)的字段作為主鍵在InnoDB中不是個好主意,因為InnoDB數(shù)據(jù)文件本身是一顆B+Tree,非單調(diào)的主鍵會造成在插入新記錄時數(shù)據(jù)文件為了維持B+Tree的特性而頻繁的分裂調(diào)整,十分低效,而使用自增字段作為主鍵則是一個很好的選擇。

    下一章將具體討論這些與索引有關(guān)的優(yōu)化策略。

    索引使用策略及優(yōu)化

    MySQL的優(yōu)化主要分為結(jié)構(gòu)優(yōu)化(Scheme optimization)和查詢優(yōu)化(Query optimization)。本章討論的高性能索引策略主要屬于結(jié)構(gòu)優(yōu)化范疇。本章的內(nèi)容完全基于上文的理論基礎(chǔ),實際上一旦理解了索引背后的機制,那么選擇高性能的策略就變成了純粹的推理,并且可以理解這些策略背后的邏輯。

    示例數(shù)據(jù)庫

    為了討論索引策略,需要一個數(shù)據(jù)量不算小的數(shù)據(jù)庫作為示例。本文選用MySQL官方文檔中提供的示例數(shù)據(jù)庫之一:employees。這個數(shù)據(jù)庫關(guān)系復(fù)雜度適中,且數(shù)據(jù)量較大。下圖是這個數(shù)據(jù)庫的E-R關(guān)系圖(引用自MySQL官方手冊):

    圖12

    MySQL官方文檔中關(guān)于此數(shù)據(jù)庫的頁面為http://dev.mysql.com/doc/employee/en/employee.html。里面詳細介紹了此數(shù)據(jù)庫,并提供了下載地址和導(dǎo)入方法,如果有興趣導(dǎo)入此數(shù)據(jù)庫到自己的MySQL可以參考文中內(nèi)容。

    最左前綴原理與相關(guān)優(yōu)化

    高效使用索引的首要條件是知道什么樣的查詢會使用到索引,這個問題和B+Tree中的“最左前綴原理”有關(guān),下面通過例子說明最左前綴原理。

    這里先說一下聯(lián)合索引的概念。在上文中,我們都是假設(shè)索引只引用了單個的列,實際上,MySQL中的索引可以以一定順序引用多個列,這種索引叫做聯(lián)合索引,一般的,一個聯(lián)合索引是一個有序元組<a1, a2, …, an>,其中各個元素均為數(shù)據(jù)表的一列,實際上要嚴格定義索引需要用到關(guān)系代數(shù),但是這里我不想討論太多關(guān)系代數(shù)的話題,因為那樣會顯得很枯燥,所以這里就不再做嚴格定義。另外,單列索引可以看成聯(lián)合索引元素數(shù)為1的特例。

    以employees.titles表為例,下面先查看其上都有哪些索引:

  • SHOW INDEX FROM employees.titles;
  • +--------+------------+----------+--------------+-------------+-----------+-------------+------+------------+
  • |Table|Non_unique|Key_name|Seq_in_index|Column_name|Collation|Cardinality|Null|Index_type|
  • +--------+------------+----------+--------------+-------------+-----------+-------------+------+------------+
  • | titles |0| PRIMARY |1| emp_no | A | NULL || BTREE |
  • | titles |0| PRIMARY |2| title | A | NULL || BTREE |
  • | titles |0| PRIMARY |3| from_date | A |443308|| BTREE |
  • | titles |1| emp_no |1| emp_no | A |443308|| BTREE |
  • +--------+------------+----------+--------------+-------------+-----------+-------------+------+------------+
  • 從結(jié)果中可以到titles表的主索引為<emp_no, title, from_date>,還有一個輔助索引<emp_no>。為了避免多個索引使事情變復(fù)雜(MySQL的SQL優(yōu)化器在多索引時行為比較復(fù)雜),這里我們將輔助索引drop掉:

  • ALTER TABLE employees.titles DROP INDEX emp_no;
  • 這樣就可以專心分析索引PRIMARY的行為了。

    情況一:全列匹配。

  • EXPLAIN SELECT * FROM employees.titles WHERE emp_no='10001' AND title='Senior Engineer' AND from_date='1986-06-26';
  • +----+-------------+--------+-------+---------------+---------+---------+-------------------+------+-------+
  • | id | select_type | table | type | possible_keys | key | key_len |ref| rows |Extra|
  • +----+-------------+--------+-------+---------------+---------+---------+-------------------+------+-------+
  • |1| SIMPLE | titles |const| PRIMARY | PRIMARY |59|const,const,const|1||
  • +----+-------------+--------+-------+---------------+---------+---------+-------------------+------+-------+
  • 很明顯,當(dāng)按照索引中所有列進行精確匹配(這里精確匹配指“=”或“IN”匹配)時,索引可以被用到。這里有一點需要注意,理論上索引對順序是敏感的,但是由于MySQL的查詢優(yōu)化器會自動調(diào)整where子句的條件順序以使用適合的索引,例如我們將where中的條件順序顛倒:

  • EXPLAIN SELECT * FROM employees.titles WHERE from_date='1986-06-26' AND emp_no='10001' AND title='Senior Engineer';
  • +----+-------------+--------+-------+---------------+---------+---------+-------------------+------+-------+
  • | id | select_type | table | type | possible_keys | key | key_len |ref| rows |Extra|
  • +----+-------------+--------+-------+---------------+---------+---------+-------------------+------+-------+
  • |1| SIMPLE | titles |const| PRIMARY | PRIMARY |59|const,const,const|1||
  • +----+-------------+--------+-------+---------------+---------+---------+-------------------+------+-------+
  • 效果是一樣的。

    情況二:最左前綴匹配。

  • EXPLAIN SELECT * FROM employees.titles WHERE emp_no='10001';
  • +----+-------------+--------+------+---------------+---------+---------+-------+------+-------+
  • | id | select_type | table | type | possible_keys | key | key_len |ref| rows |Extra|
  • +----+-------------+--------+------+---------------+---------+---------+-------+------+-------+
  • |1| SIMPLE | titles |ref| PRIMARY | PRIMARY |4|const|1||
  • +----+-------------+--------+------+---------------+---------+---------+-------+------+-------+
  • 當(dāng)查詢條件精確匹配索引的左邊連續(xù)一個或幾個列時,如<emp_no>或<emp_no, title>,所以可以被用到,但是只能用到一部分,即條件所組成的最左前綴。上面的查詢從分析結(jié)果看用到了PRIMARY索引,但是key_len為4,說明只用到了索引的第一列前綴。

    情況三:查詢條件用到了索引中列的精確匹配,但是中間某個條件未提供。

  • EXPLAIN SELECT * FROM employees.titles WHERE emp_no='10001' AND from_date='1986-06-26';
  • +----+-------------+--------+------+---------------+---------+---------+-------+------+-------------+
  • | id | select_type | table | type | possible_keys | key | key_len |ref| rows |Extra|
  • +----+-------------+--------+------+---------------+---------+---------+-------+------+-------------+
  • |1| SIMPLE | titles |ref| PRIMARY | PRIMARY |4|const|1|Usingwhere|
  • +----+-------------+--------+------+---------------+---------+---------+-------+------+-------------+
  • 此時索引使用情況和情況二相同,因為title未提供,所以查詢只用到了索引的第一列,而后面的from_date雖然也在索引中,但是由于title不存在而無法和左前綴連接,因此需要對結(jié)果進行掃描過濾from_date(這里由于emp_no唯一,所以不存在掃描)。如果想讓from_date也使用索引而不是where過濾,可以增加一個輔助索引<emp_no, from_date>,此時上面的查詢會使用這個索引。除此之外,還可以使用一種稱之為“隔離列”的優(yōu)化方法,將emp_no與from_date之間的“坑”填上。

    首先我們看下title一共有幾種不同的值:

  • SELECT DISTINCT(title) FROM employees.titles;
  • +--------------------+
  • | title |
  • +--------------------+
  • |SeniorEngineer|
  • |Staff|
  • |Engineer|
  • |SeniorStaff|
  • |AssistantEngineer|
  • |TechniqueLeader|
  • |Manager|
  • +--------------------+
  • 只有7種。在這種成為“坑”的列值比較少的情況下,可以考慮用“IN”來填補這個“坑”從而形成最左前綴:

  • EXPLAIN SELECT * FROM employees.titles
  • WHERE emp_no='10001'
  • AND title IN ('Senior Engineer','Staff','Engineer','Senior Staff','Assistant Engineer','Technique Leader','Manager')
  • AND from_date='1986-06-26';
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • | id | select_type | table | type | possible_keys | key | key_len |ref| rows |Extra|
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • |1| SIMPLE | titles | range | PRIMARY | PRIMARY |59| NULL |7|Usingwhere|
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • 這次key_len為59,說明索引被用全了,但是從type和rows看出IN實際上執(zhí)行了一個range查詢,這里檢查了7個key??聪聝煞N查詢的性能比較:

  • SHOW PROFILES;
  • +----------+------------+-------------------------------------------------------------------------------+
  • |Query_ID|Duration|Query|
  • +----------+------------+-------------------------------------------------------------------------------+
  • |10|0.00058000| SELECT * FROM employees.titles WHERE emp_no='10001' AND from_date='1986-06-26'|
  • |11|0.00052500| SELECT * FROM employees.titles WHERE emp_no='10001' AND title IN ...|
  • +----------+------------+-------------------------------------------------------------------------------+
  • “填坑”后性能提升了一點。如果經(jīng)過emp_no篩選后余下很多數(shù)據(jù),則后者性能優(yōu)勢會更加明顯。當(dāng)然,如果title的值很多,用填坑就不合適了,必須建立輔助索引。

    情況四:查詢條件沒有指定索引第一列。

  • EXPLAIN SELECT * FROM employees.titles WHERE from_date='1986-06-26';
  • +----+-------------+--------+------+---------------+------+---------+------+--------+-------------+
  • | id | select_type | table | type | possible_keys | key | key_len |ref| rows |Extra|
  • +----+-------------+--------+------+---------------+------+---------+------+--------+-------------+
  • |1| SIMPLE | titles | ALL | NULL | NULL | NULL | NULL |443308|Usingwhere|
  • +----+-------------+--------+------+---------------+------+---------+------+--------+-------------+
  • 由于不是最左前綴,索引這樣的查詢顯然用不到索引。

    情況五:匹配某列的前綴字符串。

  • EXPLAIN SELECT * FROM employees.titles WHERE emp_no='10001' AND title LIKE 'Senior%';
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • | id | select_type | table | type | possible_keys | key | key_len |ref| rows |Extra|
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • |1| SIMPLE | titles | range | PRIMARY | PRIMARY |56| NULL |1|Usingwhere|
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • 此時可以用到索引,但是如果通配符不是只出現(xiàn)在末尾,則無法使用索引。(原文表述有誤,如果通配符%不出現(xiàn)在開頭,則可以用到索引,但根據(jù)具體情況不同可能只會用其中一個前綴)

    情況六:范圍查詢。

  • EXPLAIN SELECT * FROM employees.titles WHERE emp_no <'10010'and title='Senior Engineer';
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • | id | select_type | table | type | possible_keys | key | key_len |ref| rows |Extra|
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • |1| SIMPLE | titles | range | PRIMARY | PRIMARY |4| NULL |16|Usingwhere|
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • 范圍列可以用到索引(必須是最左前綴),但是范圍列后面的列無法用到索引。同時,索引最多用于一個范圍列,因此如果查詢條件中有兩個范圍列則無法全用到索引。

  • EXPLAIN SELECT * FROM employees.titles
  • WHERE emp_no <'10010'
  • AND title='Senior Engineer'
  • AND from_date BETWEEN '1986-01-01' AND '1986-12-31';
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • | id | select_type | table | type | possible_keys | key | key_len |ref| rows |Extra|
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • |1| SIMPLE | titles | range | PRIMARY | PRIMARY |4| NULL |16|Usingwhere|
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • 可以看到索引對第二個范圍索引無能為力。這里特別要說明MySQL一個有意思的地方,那就是僅用explain可能無法區(qū)分范圍索引和多值匹配,因為在type中這兩者都顯示為range。同時,用了“between”并不意味著就是范圍查詢,例如下面的查詢:

  • EXPLAIN SELECT * FROM employees.titles
  • WHERE emp_no BETWEEN '10001' AND '10010'
  • AND title='Senior Engineer'
  • AND from_date BETWEEN '1986-01-01' AND '1986-12-31';
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • | id | select_type | table | type | possible_keys | key | key_len |ref| rows |Extra|
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • |1| SIMPLE | titles | range | PRIMARY | PRIMARY |59| NULL |16|Usingwhere|
  • +----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+
  • 看起來是用了兩個范圍查詢,但作用于emp_no上的“BETWEEN”實際上相當(dāng)于“IN”,也就是說emp_no實際是多值精確匹配??梢钥吹竭@個查詢用到了索引全部三個列。因此在MySQL中要謹慎地區(qū)分多值匹配和范圍匹配,否則會對MySQL的行為產(chǎn)生困惑。

    情況七:查詢條件中含有函數(shù)或表達式。

    很不幸,如果查詢條件中含有函數(shù)或表達式,則MySQL不會為這列使用索引(雖然某些在數(shù)學(xué)意義上可以使用)。例如:

  • EXPLAIN SELECT * FROM employees.titles WHERE emp_no='10001' AND left(title,6)='Senior';
  • +----+-------------+--------+------+---------------+---------+---------+-------+------+-------------+
  • | id | select_type | table | type | possible_keys | key | key_len |ref| rows |Extra|
  • +----+-------------+--------+------+---------------+---------+---------+-------+------+-------------+
  • |1| SIMPLE | titles |ref| PRIMARY | PRIMARY |4|const|1|Usingwhere|
  • +----+-------------+--------+------+---------------+---------+---------+-------+------+-------------+
  • 雖然這個查詢和情況五中功能相同,但是由于使用了函數(shù)left,則無法為title列應(yīng)用索引,而情況五中用LIKE則可以。再如:

    ?

  • EXPLAIN SELECT * FROM employees.titles WHERE emp_no -1='10000';
  • +----+-------------+--------+------+---------------+------+---------+------+--------+-------------+
  • | id | select_type | table | type | possible_keys | key | key_len |ref| rows |Extra|
  • +----+-------------+--------+------+---------------+------+---------+------+--------+-------------+
  • |1| SIMPLE | titles | ALL | NULL | NULL | NULL | NULL |443308|Usingwhere|
  • +----+-------------+--------+------+---------------+------+---------+------+--------+-------------+
  • 顯然這個查詢等價于查詢emp_no為10001的函數(shù),但是由于查詢條件是一個表達式,MySQL無法為其使用索引??磥鞰ySQL還沒有智能到自動優(yōu)化常量表達式的程度,因此在寫查詢語句時盡量避免表達式出現(xiàn)在查詢中,而是先手工私下代數(shù)運算,轉(zhuǎn)換為無表達式的查詢語句。

    索引選擇性與前綴索引

    既然索引可以加快查詢速度,那么是不是只要是查詢語句需要,就建上索引?答案是否定的。因為索引雖然加快了查詢速度,但索引也是有代價的:索引文件本身要消耗存儲空間,同時索引會加重插入、刪除和修改記錄時的負擔(dān),另外,MySQL在運行時也要消耗資源維護索引,因此索引并不是越多越好。一般兩種情況下不建議建索引。

    第一種情況是表記錄比較少,例如一兩千條甚至只有幾百條記錄的表,沒必要建索引,讓查詢做全表掃描就好了。至于多少條記錄才算多,這個個人有個人的看法,我個人的經(jīng)驗是以2000作為分界線,記錄數(shù)不超過 2000可以考慮不建索引,超過2000條可以酌情考慮索引。

    另一種不建議建索引的情況是索引的選擇性較低。所謂索引的選擇性(Selectivity),是指不重復(fù)的索引值(也叫基數(shù),Cardinality)與表記錄數(shù)(#T)的比值:

    Index Selectivity = Cardinality / #T

    顯然選擇性的取值范圍為(0, 1],選擇性越高的索引價值越大,這是由B+Tree的性質(zhì)決定的。例如,上文用到的employees.titles表,如果title字段經(jīng)常被單獨查詢,是否需要建索引,我們看一下它的選擇性:

  • SELECT count(DISTINCT(title))/count(*) AS Selectivity FROM employees.titles;
  • +-------------+
  • |Selectivity|
  • +-------------+
  • |0.0000|
  • +-------------+
  • title的選擇性不足0.0001(精確值為0.00001579),所以實在沒有什么必要為其單獨建索引。

    有一種與索引選擇性有關(guān)的索引優(yōu)化策略叫做前綴索引,就是用列的前綴代替整個列作為索引key,當(dāng)前綴長度合適時,可以做到既使得前綴索引的選擇性接近全列索引,同時因為索引key變短而減少了索引文件的大小和維護開銷。下面以employees.employees表為例介紹前綴索引的選擇和使用。

    從圖12可以看到employees表只有一個索引<emp_no>,那么如果我們想按名字搜索一個人,就只能全表掃描了:

  • EXPLAIN SELECT * FROM employees.employees WHERE first_name='Eric' AND last_name='Anido';
  • +----+-------------+-----------+------+---------------+------+---------+------+--------+-------------+
  • | id | select_type | table | type | possible_keys | key | key_len |ref| rows |Extra|
  • +----+-------------+-----------+------+---------------+------+---------+------+--------+-------------+
  • |1| SIMPLE | employees | ALL | NULL | NULL | NULL | NULL |300024|Usingwhere|
  • +----+-------------+-----------+------+---------------+------+---------+------+--------+-------------+
  • 如果頻繁按名字搜索員工,這樣顯然效率很低,因此我們可以考慮建索引。有兩種選擇,建<first_name>或<first_name, last_name>,看下兩個索引的選擇性:

  • SELECT count(DISTINCT(first_name))/count(*) AS Selectivity FROM employees.employees;
  • +-------------+
  • |Selectivity|
  • +-------------+
  • |0.0042|
  • +-------------+
  • SELECT count(DISTINCT(concat(first_name, last_name)))/count(*) AS Selectivity FROM employees.employees;
  • +-------------+
  • |Selectivity|
  • +-------------+
  • |0.9313|
  • +-------------+
  • <first_name>顯然選擇性太低,<first_name, last_name>選擇性很好,但是first_name和last_name加起來長度為30,有沒有兼顧長度和選擇性的辦法?可以考慮用first_name和last_name的前幾個字符建立索引,例如<first_name, left(last_name, 3)>,看看其選擇性:

  • SELECT count(DISTINCT(concat(first_name, left(last_name,3))))/count(*) AS Selectivity FROM employees.employees;
  • +-------------+
  • |Selectivity|
  • +-------------+
  • |0.7879|
  • +-------------+
  • 選擇性還不錯,但離0.9313還是有點距離,那么把last_name前綴加到4:

  • SELECT count(DISTINCT(concat(first_name, left(last_name,4))))/count(*) AS Selectivity FROM employees.employees;
  • +-------------+
  • |Selectivity|
  • +-------------+
  • |0.9007|
  • +-------------+
  • 這時選擇性已經(jīng)很理想了,而這個索引的長度只有18,比<first_name, last_name>短了接近一半,我們把這個前綴索引 建上:

  • ALTER TABLE employees.employees
  • ADD INDEX `first_name_last_name4`(first_name, last_name(4));
  • 此時再執(zhí)行一遍按名字查詢,比較分析一下與建索引前的結(jié)果:

  • SHOW PROFILES;
  • +----------+------------+---------------------------------------------------------------------------------+
  • |Query_ID|Duration|Query|
  • +----------+------------+---------------------------------------------------------------------------------+
  • |87|0.11941700| SELECT * FROM employees.employees WHERE first_name='Eric' AND last_name='Anido'|
  • |90|0.00092400| SELECT * FROM employees.employees WHERE first_name='Eric' AND last_name='Anido'|
  • +----------+------------+---------------------------------------------------------------------------------+
  • 性能的提升是顯著的,查詢速度提高了120多倍。

    前綴索引兼顧索引大小和查詢速度,但是其缺點是不能用于ORDER BY和GROUP BY操作,也不能用于Covering index(即當(dāng)索引本身包含查詢所需全部數(shù)據(jù)時,不再訪問數(shù)據(jù)文件本身)。

    InnoDB的主鍵選擇與插入優(yōu)化

    在使用InnoDB存儲引擎時,如果沒有特別的需要,請永遠使用一個與業(yè)務(wù)無關(guān)的自增字段作為主鍵。

    經(jīng)??吹接刑踊虿┛陀懻撝麈I選擇問題,有人建議使用業(yè)務(wù)無關(guān)的自增主鍵,有人覺得沒有必要,完全可以使用如學(xué)號或身份證號這種唯一字段作為主鍵。不論支持哪種論點,大多數(shù)論據(jù)都是業(yè)務(wù)層面的。如果從數(shù)據(jù)庫索引優(yōu)化角度看,使用InnoDB引擎而不使用自增主鍵絕對是一個糟糕的主意。

    上文討論過InnoDB的索引實現(xiàn),InnoDB使用聚集索引,數(shù)據(jù)記錄本身被存于主索引(一顆B+Tree)的葉子節(jié)點上。這就要求同一個葉子節(jié)點內(nèi)(大小為一個內(nèi)存頁或磁盤頁)的各條數(shù)據(jù)記錄按主鍵順序存放,因此每當(dāng)有一條新的記錄插入時,MySQL會根據(jù)其主鍵將其插入適當(dāng)?shù)墓?jié)點和位置,如果頁面達到裝載因子(InnoDB默認為15/16),則開辟一個新的頁(節(jié)點)。

    如果表使用自增主鍵,那么每次插入新的記錄,記錄就會順序添加到當(dāng)前索引節(jié)點的后續(xù)位置,當(dāng)一頁寫滿,就會自動開辟一個新的頁。如下圖所示:

    圖13

    這樣就會形成一個緊湊的索引結(jié)構(gòu),近似順序填滿。由于每次插入時也不需要移動已有數(shù)據(jù),因此效率很高,也不會增加很多開銷在維護索引上。

    如果使用非自增主鍵(如果身份證號或?qū)W號等),由于每次插入主鍵的值近似于隨機,因此每次新紀(jì)錄都要被插到現(xiàn)有索引頁得中間某個位置:

    圖14

    此時MySQL不得不為了將新記錄插到合適位置而移動數(shù)據(jù),甚至目標(biāo)頁面可能已經(jīng)被回寫到磁盤上而從緩存中清掉,此時又要從磁盤上讀回來,這增加了很多開銷,同時頻繁的移動、分頁操作造成了大量的碎片,得到了不夠緊湊的索引結(jié)構(gòu),后續(xù)不得不通過OPTIMIZE TABLE來重建表并優(yōu)化填充頁面。

    因此,只要可以,請盡量在InnoDB上采用自增字段做主鍵。

    后記

    這篇文章斷斷續(xù)續(xù)寫了半個月,主要內(nèi)容就是上面這些了。不可否認,這篇文章在一定程度上有紙上談兵之嫌,因為我本人對MySQL的使用屬于菜鳥級別,更沒有太多數(shù)據(jù)庫調(diào)優(yōu)的經(jīng)驗,在這里大談數(shù)據(jù)庫索引調(diào)優(yōu)有點大言不慚。就當(dāng)是我個人的一篇學(xué)習(xí)筆記了。

    其實數(shù)據(jù)庫索引調(diào)優(yōu)是一項技術(shù)活,不能僅僅靠理論,因為實際情況千變?nèi)f化,而且MySQL本身存在很復(fù)雜的機制,如查詢優(yōu)化策略和各種引擎的實現(xiàn)差異等都會使情況變得更加復(fù)雜。但同時這些理論是索引調(diào)優(yōu)的基礎(chǔ),只有在明白理論的基礎(chǔ)上,才能對調(diào)優(yōu)策略進行合理推斷并了解其背后的機制,然后結(jié)合實踐中不斷的實驗和摸索,從而真正達到高效使用MySQL索引的目的。

    另外,MySQL索引及其優(yōu)化涵蓋范圍非常廣,本文只是涉及到其中一部分。如與排序(ORDER BY)相關(guān)的索引優(yōu)化及覆蓋索引(Covering index)的話題本文并未涉及,同時除B-Tree索引外MySQL還根據(jù)不同引擎支持的哈希索引、全文索引等等本文也并未涉及。如果有機會,希望再對本文未涉及的部分進行補充吧。

    參考文獻

    [1] Baron Scbwartz等 著,王小東等 譯;高性能MySQL(High Performance MySQL);電子工業(yè)出版社,2010

    [2] Michael Kofler 著,楊曉云等 譯;MySQL5權(quán)威指南(The Definitive Guide to MySQL5);人民郵電出版社,2006

    [3] 姜承堯 著;MySQL技術(shù)內(nèi)幕-InnoDB存儲引擎;機械工業(yè)出版社,2011

    [4] D Comer, Ubiquitous B-tree; ACM Computing Surveys (CSUR), 1979

    [5] Codd, E. F. (1970). "A relational model of data for large shared data banks". Communications of the ACM, , Vol. 13, No. 6, pp. 377-387

    [6] MySQL5.1參考手冊 -?http://dev.mysql.com/doc/refman/5.1/zh/index.html

    轉(zhuǎn)載于:https://www.cnblogs.com/davidwang456/p/3551800.html

    《新程序員》:云原生和全面數(shù)字化實踐50位技術(shù)專家共同創(chuàng)作,文字、視頻、音頻交互閱讀

    總結(jié)

    以上是生活随笔為你收集整理的MySQL索引背后的数据结构及算法原理--转的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

    五月开心婷婷网 | 亚洲 中文 在线 精品 | 在线观看亚洲专区 | 麻豆91精品视频 | 久草在线欧美 | 免费在线观看一级片 | 亚洲 欧洲 国产 日本 综合 | 香蕉97视频观看在线观看 | 成人a毛片 | 国产精品久久久久一区二区 | 日韩最新理论电影 | 特级片免费看 | 亚洲精品国偷拍自产在线观看蜜桃 | 超碰免费成人 | 最近中文字幕高清字幕在线视频 | 国产黄免费在线观看 | 曰本三级在线 | 天天av天天| 精精国产xxxx视频在线播放 | 人人草在线观看 | 99国产精品久久久久老师 | 色综合婷婷 | 久久黄色a级片 | 久久精品国产第一区二区三区 | 久久视频这里有久久精品视频11 | 一级黄色片在线播放 | 91av亚洲 | 99免费观看视频 | 一区二区三区在线视频观看58 | 天天看天天操 | 网站你懂的 | 日韩久久激情 | 福利视频午夜 | 日韩一区二区三免费高清在线观看 | 日韩成人精品一区二区三区 | 久久久 精品 | 一级a性色生活片久久毛片波多野 | 国产又粗又猛又色又黄网站 | 免费热情视频 | 国产 字幕 制服 中文 在线 | 999久久久精品视频 日韩高清www | 91日韩免费 | 欧美五月婷婷 | 麻豆成人在线观看 | 日本性生活一级片 | 国产小视频在线免费观看视频 | 91重口视频 | 国产成人在线看 | 久久午夜精品视频 | 欧美激情xxxx性bbbb | 在线之家免费在线观看电影 | 日韩免费观看高清 | 99久久超碰中文字幕伊人 | 免费看特级毛片 | 玖玖视频精品 | 99视频免费播放 | 精品字幕 | 91久久久久久久 | 国产一二三区在线观看 | 公开超碰在线 | 国产精品综合在线观看 | 播五月综合 | 欧美色888 | 免费日韩高清 | 337p日本欧洲亚洲大胆裸体艺术 | 精品国产综合区久久久久久 | 日韩精品一区二区免费视频 | 欧美高清视频不卡网 | 亚洲高清视频在线观看免费 | 国产品久精国精产拍 | 久久久久国产a免费观看rela | 欧美最新另类人妖 | 午夜av日韩 | 手机在线看永久av片免费 | 涩av在线 | 天天天天色射综合 | 婷婷综合久久 | 区一区二在线 | 久久综合激情 | 国产色在线观看 | 成人电影毛片 | 91精品久久久久久久91蜜桃 | 色干综合| 黄色成人91| 天天干 夜夜操 | 国产精品九九视频 | 成人免费观看视频网站 | 五月激情丁香 | 欧美资源在线观看 | 国产系列精品av | 日b黄色片| 亚洲aⅴ一区二区三区 | 亚洲男男gaygayxxxgv | 亚洲网站在线看 | 久久综合狠狠综合 | 久久免费看毛片 | 97超碰总站 | 日韩中文字幕在线看 | 涩涩伊人 | av色一区 | 91av成人 | 国产色资源 | 黄色毛片视频 | 国产免费久久 | 人人添人人澡 | 日韩午夜精品 | 国产美女在线观看 | 日韩视频一| 久久不卡国产精品一区二区 | 国产麻豆剧果冻传媒视频播放量 | 97香蕉视频 | 亚洲免费专区 | 日本aaaa级毛片在线看 | 91久久黄色 | 久爱精品在线 | av888av.com| 国产精品九九视频 | 麻豆视传媒官网免费观看 | av天天干| 天天射天天射 | 久久久国产高清 | 国产99区 | 波多野结衣一区二区 | 国产一卡久久电影永久 | 亚洲麻豆精品 | 成人av片免费看 | 国产亚洲精品久久久久久久久久 | 九九九在线观看视频 | 开心激情综合网 | 欧美极品久久 | 久久99国产精品久久 | 99色99| 国产精品手机视频 | 国产一二区视频 | 麻豆久久一区二区 | 国产又粗又猛又爽 | 亚洲精品久久视频 | 日韩av影片在线观看 | 99视频 | 一级a毛片高清视频 | 五月天久久 | 成年人免费电影在线观看 | 亚洲人成精品久久久久 | 精品伦理一区二区三区 | 六月婷婷网 | 高清在线一区 | 天天天天射 | www.狠狠操.com| 97电影院网| 午夜av影院 | 在线成人免费电影 | 久久国产视频网 | 天天天天天干 | 色婷婷88av视频一二三区 | 日韩免费视频在线观看 | 激情欧美一区二区免费视频 | 97操操操| 超级碰碰碰视频 | 日本一区二区三区视频在线播放 | 亚洲午夜久久久久久久久 | 天天操天天射天天 | 激情综合网在线观看 | 日韩一级理论片 | 日韩午夜精品 | 精品久久久久久久久亚洲 | 久久9视频| 国产又粗又猛又黄 | 亚洲精品777 | 狠狠躁天天躁 | av资源免费在线观看 | 欧美日本日韩aⅴ在线视频 插插插色综合 | 在线观看亚洲免费视频 | 人人插人人爱 | 中文字幕第一页在线 | 中文字幕免费高清av | 成人免费中文字幕 | 久久字幕精品一区 | 91视频电影 | 菠萝菠萝在线精品视频 | 亚洲精品视频在 | 国产 字幕 制服 中文 在线 | 亚洲精品久久久蜜桃直播 | 亚洲国产免费看 | 久艹视频免费观看 | 亚洲 欧美日韩 国产 中文 | 欧美精品首页 | 五月婷婷.com | 丁香久久| 狠狠色丁香久久婷婷综合五月 | 91麻豆精品国产91久久久更新时间 | 中文字幕制服丝袜av久久 | av 一区 二区 久久 | 亚洲va欧美va人人爽 | 成人在线视频免费 | 成人性生交大片免费看中文网站 | 在线观看成人一级片 | 97色婷婷人人爽人人 | 国产资源中文字幕 | 免费观看国产成人 | 国产免费一区二区三区网站免费 | 免费麻豆视频 | 久久艹综合 | 97视频网站 | 国产在线观看午夜 | 成人资源在线观看 | 91系列在线 | 欧美色综合天天久久综合精品 | 人人盈棋牌| 91人人爽人人爽人人精88v | 日韩视频在线播放 | 久久综合成人 | 日韩一级片大全 | 在线观看视频你懂得 | 久久久精品欧美 | 欧美另类交在线观看 | 国产精品一区久久久久 | 在线免费观看av网站 | 99热高清 | 国产精品99久久久久久小说 | 国产精品成人国产乱 | 91大神一区二区三区 | 国产精品久久综合 | 中文字幕日韩一区二区三区不卡 | 69国产盗摄一区二区三区五区 | 精品一二三四五区 | 国产在线精品区 | wwwwww色| 69视频在线 | 久久综合99 | 欧美另类调教 | japanesexxxhd奶水 91在线精品一区二区 | 玖玖爱免费视频 | 国产91影院 | av免费看av | 日韩毛片在线播放 | 国产伦精品一区二区三区免费 | 久久精品99国产精品日本 | 女人高潮一级片 | 久久久久免费精品国产 | 亚洲精品在线视频播放 | 五月天中文字幕 | 久草香蕉在线视频 | 久久五月情影视 | 韩日精品在线 | 一级性视频 | 91成版人在线观看入口 | 久久精品视频在线看 | 五月天激情视频在线观看 | 精品一区二区av | 国产在线一卡 | 97视频在线看 | 日b视频在线观看网址 | 亚洲激情综合 | 国产精品都在这里 | 中文成人字幕 | 日韩在线观看不卡 | 国产毛片久久久 | 91av亚洲 | 亚洲国产精品影院 | 91成人网在线 | 最近中文字幕完整视频高清1 | 在线免费亚洲 | 久久99国产精品免费网站 | 国产高清专区 | 欧美特一级 | 深爱婷婷激情 | 久久精品国产v日韩v亚洲 | 97在线看| 国内精品国产三级国产aⅴ久 | 中文字幕一区av | 亚洲涩涩涩 | 99精品视频免费观看视频 | 欧美日本在线观看视频 | 日本久久片 | 欧美精品成人在线 | 午夜免费福利视频 | 99国产精品 | 在线观看色网站 | av先锋中文字幕 | 久久亚洲精品国产亚洲老地址 | 国产在线第三页 | 在线观看91精品视频 | 丁香婷婷激情 | 97精品久久人人爽人人爽 | 九九热精品视频在线播放 | 在线看欧美 | 日韩二区在线播放 | 国产中文字幕一区二区三区 | av在线网站免费观看 | 欧美精品久久久久久 | 亚洲成 人精品 | 国产一级小视频 | 午夜视频一区二区 | 天天干天天做 | 91中文字幕在线 | 日韩福利在线观看 | 国产高清在线免费视频 | 主播av在线 | 伊人天天干 | 亚洲精品视频播放 | 日韩精品久久久久久中文字幕8 | 久久爱资源网 | av一本久道久久波多野结衣 | 国产午夜三级一区二区三桃花影视 | 日韩免费视频线观看 | 在线免费高清视频 | 欧美一级片免费播放 | 久久久久国产一区二区三区 | 中文字幕日韩一区二区三区不卡 | 亚洲免费国产视频 | 日韩乱理 | 日本在线h | av官网在线 | 久久久久久久久福利 | 99re中文字幕 | 69亚洲乱 | 91黄色小网站| 成人免费观看网站 | 国产a高清 | 99久久精品一区二区成人 | 麻豆av一区二区三区在线观看 | 精品国产久 | 国产在线传媒 | 日日夜夜噜噜噜 | 亚洲午夜久久久影院 | 久久不射电影网 | 在线视频亚洲 | 久久综合久久综合这里只有精品 | 日韩高清在线观看 | 天天色影院 | 久久久黄色 | 日韩精品在线看 | 日韩精品免费在线观看 | 丁香六月av | 日韩a在线| 五月天,com| 日韩av高清| 亚洲国产一区二区精品专区 | 国产精品免费一区二区三区 | 十八岁以下禁止观看的1000个网站 | 在线国产激情视频 | 国产剧情在线一区 | 国产成人在线观看 | 精品视频免费久久久看 | 99久久精品网| 日本久久中文字幕 | 久草精品资源 | 亚洲日韩精品欧美一区二区 | a天堂一码二码专区 | www.夜色.com | 99精品视频免费全部在线 | 久久99久久99久久 | 狠狠操狠狠干天天操 | 草久中文字幕 | 久久精品国产成人精品 | 国产婷婷久久 | 国产精品久久婷婷六月丁香 | 国产精品成人在线观看 | 婷婷中文字幕在线观看 | 天堂麻豆| 五月天丁香综合 | 精品在线看 | 国产精品成久久久久 | 精品久久久久久亚洲 | 久久社区视频 | 色视频在线观看免费 | 中文字幕一区二区在线播放 | 91丨九色丨91啦蝌蚪老版 | 日日爱av| 免费成人av在线看 | 91亚洲网站 | 国产精品麻豆欧美日韩ww | 日韩成人精品在线观看 | 在线99热 | 国产精品99久久久久的智能播放 | 又大又硬又黄又爽视频在线观看 | 香蕉97视频观看在线观看 | 亚洲最新视频在线 | 日韩视频一区二区在线观看 | 久久99精品久久久久蜜臀 | 久久狠狠亚洲综合 | aaa亚洲精品一二三区 | 国产精品一区二区无线 | 久久精品二区 | 日韩二区三区在线 | 激情综合啪啪 | 色婷婷免费视频 | 奇米网444| 亚洲高清视频一区二区三区 | 一级成人免费视频 | 久久艹中文字幕 | 五月婷婷狠狠 | 日韩av网页 | 日韩电影在线观看中文字幕 | 免费高清在线观看成人 | 一区免费视频 | 国产一级片毛片 | 欧美日韩国产三级 | www91在线观看 | 在线免费av网 | 久久精品99北条麻妃 | 亚洲视频久久 | 久久99亚洲精品久久久久 | www..com毛片 | 免费a级黄色毛片 | 四虎影视成人永久免费观看亚洲欧美 | 人人看人人 | 国产精品久久久久久高潮 | 国产免费久久久久 | 欧洲精品二区 | 国产免费又爽又刺激在线观看 | 在线观看韩日电影免费 | 黄色小说视频网站 | 91av成人 | 91手机电视 | 日韩免费视频 | 中文字幕免费高 | 九热精品| 在线观看www91| 久久xx视频| 色婷婷激情四射 | 美女网站在线 | 成人国产精品av | 伊人六月 | 久久精品5| 日韩h在线观看 | 亚洲人精品午夜 | 最近中文字幕国语免费av | av综合av| 九九免费观看视频 | 亚洲精品高清在线观看 | 久久99国产精品久久 | 国产不卡一区二区视频 | 精品一区电影 | 精品国产a | 天天射天天干天天操 | 久久久久久久亚洲精品 | 911亚洲精品第一 | 毛片久久久 | 国产精品久久片 | 国产.精品.日韩.另类.中文.在线.播放 | 国产成人一区二区三区久久精品 | 天天草天天色 | 久久久99久久 | 精品久久久久久亚洲综合网站 | 在线观看黄色小视频 | 日本在线观看中文字幕无线观看 | 91精品夜夜 | 日韩国产精品一区 | 国产淫片免费看 | 69视频在线 | 国产精品 亚洲精品 | 蜜臀久久99精品久久久酒店新书 | 亚洲黄在线观看 | 国产精品午夜av | 91九色蝌蚪国产 | 中文字幕资源网在线观看 | 欧美在线观看禁18 | 国产区精品 | 九色91在线视频 | 日韩午夜在线播放 | 婷婷色中文字幕 | 久久亚洲精品电影 | 伊香蕉大综综综合久久啪 | 欧美极品在线播放 | 国产黄色一级片在线 | 久久99久久99精品中文字幕 | 天堂在线成人 | 欧美日性视频 | 少妇做爰k8经典 | 精品专区一区二区 | 美腿丝袜一区二区三区 | 天天躁天天操 | 国产亚洲视频在线免费观看 | 成人中文字幕在线观看 | 国产精品福利在线观看 | 久久精品三级 | 五月天综合色激情 | 亚洲免费在线观看视频 | 久久综合狠狠 | 麻豆国产精品va在线观看不卡 | 成人激情开心网 | 日韩精品网址 | 成人免费视频网址 | 中文字幕在线高清 | 久久精品久久99精品久久 | 98精品国产自产在线观看 | 狠狠色丁香婷婷综合最新地址 | 五月婷在线视频 | 狠狠狠色丁香综合久久天下网 | 久久久久久久久免费视频 | av3级在线| 婷色在线 | 色噜噜狠狠狠狠色综合 | 免费日韩在线 | 国产色视频一区 | 99久久精品国产欧美主题曲 | 天天鲁天天干天天射 | 69国产精品视频 | 欧美aa一级| 婷婷国产视频 | 狠狠干婷婷 | 成人免费视频网站 | 深夜免费网站 | 久久涩视频 | 99精品视频免费观看 | 久久网站av | 国产免费成人 | 国产精品视频最多的网站 | 免费日韩一区二区三区 | 国产一区二区精 | 丰满少妇在线观看资源站 | 欧美日韩亚洲第一 | 日本在线视频一区二区三区 | 久草在线在线视频 | 国产精品12 | 久久免费国产电影 | 久久久久久久久久久网 | 美女视频久久久 | 三级黄色片子 | 久久婷婷亚洲 | 色综合天天色综合 | a级国产乱理伦片在线播放 久久久久国产精品一区 | 国内精品久久久久影院男同志 | 免费高清在线观看成人 | 成人精品影视 | 久久精品国产精品亚洲 | 97热久久免费频精品99 | 日韩高清不卡一区二区三区 | 黄色一级大片在线免费看国产一 | 中文av在线播放 | 久久久高清免费视频 | 久草视频在线观 | 97在线观看免费观看高清 | 黄色成品视频 | 91免费观看视频网站 | 久久99在线观看 | 在线观看激情av | 亚洲精品在线观 | 奇米影音四色 | 久久久久日本精品一区二区三区 | 青青河边草免费视频 | 999国产精品视频 | 久久综合色一综合色88 | 午夜精品久久久久久久99婷婷 | 天天躁日日躁狠狠躁av中文 | 日韩在线视频免费播放 | 91网站免费观看 | 国产高清av在线播放 | 在线免费看片 | a级免费观看 | 丁五月婷婷| 国产一区二区三区网站 | 日本中文字幕久久 | 美女在线免费观看视频 | 天天操天天干天天爽 | 激情欧美日韩一区二区 | 免费观看一级特黄欧美大片 | 91精品啪在线观看国产线免费 | 成人在线视频你懂的 | 国语自产偷拍精品视频偷 | a天堂一码二码专区 | 婷婷久月 | 香蕉视频91 | 美女黄网久久 | 视频国产| 久久一视频 | 免费开视频 | 国产色女人| 亚洲国产精品va在线看黑人 | 小草av在线播放 | 一本一本久久a久久精品综合 | 久久久久在线视频 | 国产精华国产精品 | 精品国内自产拍在线观看视频 | 欧洲av在线 | 久久毛片视频 | 亚洲经典视频 | 日本精品久久久久影院 | 成人中心免费视频 | 草久热 | 欧美日韩不卡在线 | 丁香久久久 | 久久精品国产久精国产 | 日韩欧美一区二区三区视频 | 99re亚洲国产精品 | 免费看国产一级片 | 免费黄色在线播放 | 国色天香永久免费 | 久久国产成人午夜av影院宅 | 人人藻人人澡人人爽 | 久久国产精品一区二区三区四区 | 国产偷v国产偷∨精品视频 在线草 | 一区二区视 | 久久精彩 | 一区二区激情视频 | 国产精品成久久久久三级 | 韩日精品中文字幕 | 亚洲天堂社区 | 中文字幕av免费在线观看 | 中文在线8新资源库 | 韩国精品福利一区二区三区 | 视色网站| 亚洲精品视频在 | 国产亚洲精品久久久久5区 成人h电影在线观看 | 久一久久 | 成人久久网 | 欧美精品久久久久久久久久白贞 | 一区二区三区四区精品 | 亚洲精品啊啊啊 | 亚洲精品国偷拍自产在线观看蜜桃 | 黄色资源在线观看 | 国产精品日韩在线 | 91资源在线 | 777奇米四色 | 国产精品美女久久久 | 99精品一区 | 97免费视频在线播放 | 久久超碰网 | 在线播放 日韩专区 | 亚洲综合在线一区二区三区 | 国产精品成人久久 | 91免费观看 | 国产精品区免费视频 | 成人av在线影院 | 亚洲精品午夜一区人人爽 | 欧美激情视频在线免费观看 | 日本韩国在线不卡 | 久草香蕉在线视频 | 中午字幕在线 | 亚洲欧美国产精品久久久久 | 黄色软件大全网站 | 一本一道久久a久久综合蜜桃 | 99久久精品国产观看 | 日日婷婷夜日日天干 | 黄色av网站在线免费观看 | 六月激情 | 久久成人高清 | 欧美在线观看小视频 | 成人午夜影视 | 香蕉在线观看 | 国产精品免费一区二区三区在线观看 | 性色av一区二区三区在线观看 | 国产一区二区成人 | 天天摸天天干天天操天天射 | 日韩精品视频在线观看网址 | 91精品国产91热久久久做人人 | 欧美a级在线 | 精品国产亚洲一区二区麻豆 | 国产淫片| 国产精品18久久久久vr手机版特色 | 久久久高清免费视频 | 97超碰人人澡人人爱学生 | 少妇bbbb| 91av在线精品 | 久久久91精品国产一区二区三区 | 久久久免费观看视频 | 欧美福利视频 | 国产精品久久婷婷六月丁香 | 中文字幕有码在线播放 | 亚洲国产精品va在线看黑人动漫 | 欧美精品久久久久久久亚洲调教 | 免费观看成年人视频 | 中文不卡视频 | 欧美不卡视频在线 | 亚洲aⅴ久久精品 | 婷婷五月在线视频 | 黄色片视频免费 | 四虎永久国产精品 | av福利网址导航 | 人人爽人人香蕉 | 美女黄色网在线播放 | 免费看三级网站 | 亚洲成人在线免费 | 91精品免费视频 | 手机在线免费av | 热99久久精品| 天天天色综合 | 亚洲欧美国产精品 | 天天曰天天爽 | 久久影视一区二区 | 欧日韩在线视频 | 亚洲伦理中文字幕 | 四虎国产精品成人免费影视 | 国产婷婷精品 | 日韩久久精品一区二区三区下载 | 男女拍拍免费视频 | 天堂在线一区二区 | 久久婷婷国产色一区二区三区 | 国语精品免费视频 | 日韩sese| 超碰国产人人 | 四虎精品成人免费网站 | 一区二区三区在线播放 | 草久久精品 | 久久久电影| 久久久久99999| 久久久久久久久久久久久久av | 69av网| www亚洲精品| 五月婷婷六月丁香 | 天天爱天天 | 97精品国产97久久久久久春色 | 五月激情天 | 69av久久| 精品在线观看视频 | 久久手机视频 | 又污又黄网站 | av爱干| 国产精品免费麻豆入口 | 黄色录像av | 欧美嫩草影院 | 久久99热这里只有精品国产 | 人人精久 | 国产色资源| 9999在线观看 | 超碰在线97免费 | 成人免费观看大片 | www.夜夜 | 又黄又爽又无遮挡的视频 | 91麻豆精品国产午夜天堂 | 国产999精品久久久影片官网 | 欧美性色综合网 | 精品免费视频. | 免费观看版 | 免费精品人在线二线三线 | 亚洲高清免费在线 | 欧美老人xxxx18 | 丁香婷婷激情国产高清秒播 | 狠狠狠狠狠狠狠狠干 | 国内视频1区 | 99久久99 | 久久福利剧场 | 国产免费叼嘿网站免费 | 91porny九色91啦中文 | 韩国在线一区二区 | 国产日韩欧美在线看 | 欧美日本高清视频 | 五月天久久久久 | 亚洲高清不卡av | 91视视频在线直接观看在线看网页在线看 | 亚洲精品成人免费 | 人人添人人澡人人澡人人人爽 | 日韩中文字幕免费在线播放 | 国产精品黄色影片导航在线观看 | 久久欧美精品 | 黄色大片免费播放 | 久久小视频 | 欧美视频日韩视频 | 久久免费看视频 | 不卡的av片| 九九热在线观看视频 | 亚洲成人欧美 | 99中文在线 | 国产资源在线免费观看 | 日韩性xxx| 又爽又黄在线观看 | 国产视频在线观看免费 | 日韩三级中文字幕 | 久久综合久久综合这里只有精品 | 中文字幕文字幕一区二区 | 午夜久久久精品 | av中文字幕网站 | 91看片淫黄大片一级在线观看 | 国产一区二区不卡视频 | 欧美一级欧美一级 | 成人影片免费 | 欧美另类色图 | 97在线资源 | 蜜臀久久99精品久久久无需会员 | 在线亚州 | 久久国产香蕉视频 | 欧美少妇影院 | 999国内精品永久免费视频 | 九9热这里真品2 | 亚洲精品国产区 | 日本中文乱码卡一卡二新区 | 日韩69视频| 免费下载高清毛片 | 亚洲精品欧洲精品 | 欧美人人爱 | 国产精品99久久免费黑人 | 狠狠网亚洲精品 | 成人午夜精品福利免费 | 日韩大片在线观看 | 中文字幕黄色网 | 中文字幕日韩在线播放 | www黄在线 | 九九国产视频 | 日韩高清不卡一区二区三区 | 色五婷婷 | 日韩电影在线观看中文字幕 | 亚洲精品小视频 | 久久综合给合久久狠狠色 | 久久草草影视免费网 | 国产欧美日韩一区 | 在线电影日韩 | 免费看精品久久片 | 久久久精品 | 懂色av一区二区在线播放 | 国产精品门事件 | 色黄www小说 | 中文字幕av最新 | 91精品一区二区三区久久久久久 | 中文字幕在线中文 | 日韩一级片观看 | 国产精品欧美精品 | 精品国产一区二区三区男人吃奶 | 在线视频一二区 | 婷婷成人亚洲综合国产xv88 | 18做爰免费视频网站 | 国产伦精品一区二区三区… | 国产在线毛片 | www.午夜视频 | 在线播放亚洲 | 国产成人一区二区啪在线观看 | 国内精品久久久久影院男同志 | 日韩精品免费一区二区 | 国内99视频 | 亚洲精品久久久久久久不卡四虎 | 午夜视频一区二区 | 91在线精品播放 | 九九交易行官网 | 欧美一区二区在线刺激视频 | 久久久久五月 | 国产精品久久久 | 日韩国产精品一区 | 97精品久久人人爽人人爽 | 一区 二区 精品 | 成人黄色大片在线观看 | 日韩精品专区 | 国产福利不卡视频 | 中文有码在线视频 | 国内外成人免费在线视频 | 波多野结衣视频一区二区三区 | 精品国产乱码一区二区三区在线 | 69精品久久久| 天天射天天操天天干 | 亚洲日本欧美在线 | 欧美日产在线观看 | 午夜精品婷婷 | 亚洲高清激情 | 国产精品久久久久久久妇 | 国产精品日韩久久久久 | 精品久久久免费 | 日韩视频在线一区 | 国产免费一区二区三区最新6 | 国产精品久久久久亚洲影视 | 免费成人av在线 | 成人97视频| 国产伦精品一区二区三区四区视频 | 精品国产三级a∨在线欧美 免费一级片在线观看 | 啪嗒啪嗒免费观看完整版 | 国产亚洲精品久久网站 | 丝袜网站在线观看 | 91av手机在线观看 | 国产成人精品久久二区二区 | 日韩欧美在线视频一区二区三区 | 狠狠色噜噜狠狠狠合久 | 免费的黄色的网站 | 亚洲狠狠操 | 国产黄色免费 | 中文字幕在线资源 | 毛片无卡免费无播放器 | 欧美一级黄色视屏 | 鲁一鲁影院 | 国内视频 | 99视频偷窥在线精品国自产拍 | 亚洲欧美精品在线 | 黄网站app在线观看免费视频 | 五月天婷婷免费视频 | 成人福利在线 | 久久这里只有精品视频99 | 国产精品都在这里 | 天天射天天干天天爽 | 香蕉在线视频播放网站 | 天天操,夜夜操 | www.久久久 | 在线观看日韩 | 日韩高清免费观看 | 在线观看黄色的网站 | 欧美精品久久久久久久久免 | 精品国产乱码久久久久久1区二区 | 丁香六月婷婷激情 | 亚洲国产中文字幕在线 | 亚洲人成精品久久久久 | 99精品视频免费看 | 亚洲三级在线免费观看 | 91在线观看欧美日韩 | 激情丁香5月 | 亚洲欧洲日韩在线观看 | www.日日操.com| 欧美日韩二区三区 | 婷婷丁香色 | 久久久精品一区二区三区 | 久久伊人精品一区二区三区 | 99在线热播精品免费99热 | 91porny九色91啦中文 | 国产成人av一区二区三区在线观看 | 日韩剧情| 免费在线观看一区二区三区 | 激情五月***国产精品 | 日本在线观看一区二区 | 国产你懂的在线 | 亚洲精品在线视频观看 | 亚洲精欧美一区二区精品 | 国产一级特黄毛片在线毛片 | 黄色国产在线 | 五月婷婷亚洲 | 国产精品久久久久久a | 五月色丁香 | 在线黄色毛片 | 97在线视频免费 | 91丨九色丨国产女 | 伊人久久精品久久亚洲一区 | 国语精品免费视频 | 欧美韩国日本在线 | 免费日p视频 | 五月天天色 | 国产一级黄色av | 天天操天天干天天插 | 91高清视频在线 | 亚洲三区在线 | 97超碰在线视 | 国产手机在线观看视频 | 国产精选视频 | 精品一区二区久久久久久久网站 | 最近中文字幕完整高清 | 国产精品久久一区二区三区, | 久久伊人操 | 国产小视频你懂的在线 | 中国一级片在线播放 | 免费在线观看日韩视频 | 久久不卡国产精品一区二区 | 中文在线www| 中国一级片视频 | 欧美性色网站 | 婷婷午夜 | 欧美日韩在线免费观看视频 | www.狠狠色.com| 久草精品在线观看 | 91亚洲国产 | 久久精品成人欧美大片古装 | 在线观看岛国av | 国产精品久久久久永久免费 | 亚洲黄色一级大片 | 日韩在线视频线视频免费网站 | 伊人中文在线 | 久久夜靖品 | 国产123av| 丁香六月婷婷综合 | 波多野结衣视频一区二区三区 | 久久久免费毛片 | 色干干| 在线观看视频中文字幕 | 91成人观看 | 国语久久 | 久久久久在线视频 | www.天天色.com | 一区二区三区在线免费 | 91人人澡人人爽人人精品 | 国产精品每日更新 | av不卡免费在线观看 | 欧美老人xxxx18 | 天天艹天天干天天 | 亚洲.www | 欧美极品久久 | 91精品久久久久 | 成人免费视频播放 | av在线超碰 | 91福利视频免费观看 | 99久久er热在这里只有精品66 | 亚洲永久av | 天天操天天操天天操天天操天天操 | 黄色国产区 | 美女福利视频网 | 97久久久免费福利网址 | 五月天视频网 | 91福利社区在线观看 | 日韩中文字幕免费在线播放 | 国产在线一区二区三区播放 | 日本精品在线看 | 丁香六月伊人 | 亚洲小视频在线观看 | 99精品欧美一区二区三区 | 国产精品一级在线 | 成人午夜剧场在线观看 | 日韩精品大片 | 高清不卡一区二区在线 | 日韩一区精品 | 波多野结衣视频一区 | 中文字幕视频网站 | 午夜精品久久久久久久99 | av网址aaa| 久久综合99 | 久久1区 | 日韩天堂在线观看 | 日韩不卡高清 |