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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > 数据库 >内容正文

数据库

mysql数据库算法_数据库:MySQL索引背后的数据结构及算法原理【转】

發布時間:2023/12/10 数据库 44 豆豆
生活随笔 收集整理的這篇文章主要介紹了 mysql数据库算法_数据库:MySQL索引背后的数据结构及算法原理【转】 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

原文:http://blog.codinglabs.org/articles/theory-of-mysql-index.html

摘要

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

文章主要內容分為三個部分。

第一部分主要從數據結構及算法理論層面討論MySQL數據庫索引的數理基礎。

第二部分結合MySQL數據庫中MyISAM和InnoDB數據存儲引擎中索引的架構實現討論聚集索引、非聚集索引及覆蓋索引等話題。

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

數據結構及算法基礎

索引的本質

MySQL官方對索引的定義為:索引(Index)是幫助MySQL高效獲取數據的數據結構。提取句子主干,就可以得到索引的本質:索引是數據結構。

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

看一個例子:

圖1

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

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

B-Tree和B+Tree

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

B-Tree

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

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

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

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

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

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

key和指針互相間隔,節點兩端是指針。

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

所有節點組成樹結構。

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

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

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

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

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

圖2

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

BTree_Search(node, key) {if(node == null) return null;

foreach(node.key)

{if(node.key[i] == key) returnnode.data[i];if(node.key[i] > key) return BTree_Search(point[i]->node);

}return BTree_Search(point[i+1]->node);

}

data= BTree_Search(root, my_key);

關于B-Tree有一系列有趣的性質,例如一個度為d的B-Tree,設其索引N個key,則其樹高h的上限為logd((N+1)/2),檢索一個key,其查找節點個數的漸進復雜度為O(logdN)。從這點可以看出,B-Tree是一個非常有效率的索引數據結構。

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

B+Tree

B-Tree有許多變種,其中最常見的是B+Tree,例如MySQL就普遍使用B+Tree實現其索引結構。

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

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

內節點不存儲data,只存儲key;葉子節點不存儲指針。

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

圖3

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

一般來說,B+Tree比B-Tree更適合實現外存儲索引結構,具體原因與外存儲器原理及計算機存取原理有關,將在下面討論。

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

一般在數據庫系統或文件系統中使用的B+Tree結構都在經典B+Tree的基礎上進行了優化,增加了順序訪問指針。

圖4

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

這一節對B-Tree和B+Tree進行了一個簡單的介紹,下一節結合存儲器存取原理介紹為什么目前B+Tree是數據庫系統實現索引的首選數據結構。

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

上文說過,紅黑樹等數據結構也可以用來實現索引,但是文件系統及數據庫系統普遍采用B-/+Tree作為索引結構,這一節將結合計算機組成原理相關知識討論B-/+Tree作為索引的理論基礎。

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

主存存取原理

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

圖5

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

主存的存取過程如下:

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

寫主存的過程類似,系統將要寫入單元地址和數據分別放在地址總線和數據總線上,主存讀取兩個總線的內容,做相應的寫操作。

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

磁盤存取原理

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

圖6是磁盤的整體結構示意圖。

圖6

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

圖7是磁盤結構的示意圖。

圖7

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

當需要從磁盤讀取數據時,系統會將數據邏輯地址傳給磁盤,磁盤的控制電路按照尋址邏輯將邏輯地址翻譯成物理地址,即確定要讀的數據在哪個磁道,哪個扇區。為了讀取這個扇區的數據,需要將磁頭放到這個扇區上方,為了實現這一點,磁頭需要移動對準相應磁道,這個過程叫做尋道,所耗費時間叫做尋道時間,然后磁盤旋轉將目標扇區旋轉到磁頭下,這個過程耗費的時間叫做旋轉時間。

局部性原理與磁盤預讀

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

當一個數據被用到時,其附近的數據也通常會馬上被使用。

程序運行期間所需要的數據通常比較集中。

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

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

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

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

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

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

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

綜上所述,用B-Tree作為索引結構效率是非常高的。

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

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

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

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

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

MySQL索引實現

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

MyISAM索引實現

MyISAM引擎使用B+Tree作為索引結構,葉節點的data域存放的是數據記錄的地址。下圖是MyISAM索引的原理圖:

圖8

這里設表一共有三列,假設我們以Col1為主鍵,則圖8是一個MyISAM表的主索引(Primary key)示意。可以看出MyISAM的索引文件僅僅保存數據記錄的地址。在MyISAM中,主索引和輔助索引(Secondary key)在結構上沒有任何區別,只是主索引要求key是唯一的,而輔助索引的key可以重復。如果我們在Col2上建立一個輔助索引,則此索引的結構如下圖所示:

圖9

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

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

InnoDB索引實現

雖然InnoDB也使用B+Tree作為索引結構,但具體實現方式卻與MyISAM截然不同。

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

圖10

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

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

圖11

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

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

下一章將具體討論這些與索引有關的優化策略。

索引使用策略及優化

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

示例數據庫

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

圖12

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

最左前綴原理與相關優化

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

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

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

SHOW INDEX FROMemployees.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 |

+--------+------------+----------+--------------+-------------+-----------+-------------+------+------------+

從結果中可以到titles表的主索引為,還有一個輔助索引。為了避免多個索引使事情變復雜(MySQL的SQL優化器在多索引時行為比較復雜),這里我們將輔助索引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 | |

+----+-------------+--------+-------+---------------+---------+---------+-------------------+------+-------+

很明顯,當按照索引中所有列進行精確匹配(這里精確匹配指“=”或“IN”匹配)時,索引可以被用到。這里有一點需要注意,理論上索引對順序是敏感的,但是由于MySQL的查詢優化器會自動調整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 | |

+----+-------------+--------+------+---------------+---------+---------+-------+------+-------+

當查詢條件精確匹配索引的左邊連續一個或幾個列時,如或,所以可以被用到,但是只能用到一部分,即條件所組成的最左前綴。上面的查詢從分析結果看用到了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 | Using where |

+----+-------------+--------+------+---------------+---------+---------+-------+------+-------------+

此時索引使用情況和情況二相同,因為title未提供,所以查詢只用到了索引的第一列,而后面的from_date雖然也在索引中,但是由于title不存在而無法和左前綴連接,因此需要對結果進行掃描過濾from_date(這里由于emp_no唯一,所以不存在掃描)。如果想讓from_date也使用索引而不是where過濾,可以增加一個輔助索引,此時上面的查詢會使用這個索引。除此之外,還可以使用一種稱之為“隔離列”的優化方法,將emp_no與from_date之間的“坑”填上。

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

SELECT DISTINCT(title) FROMemployees.titles;+--------------------+

| title |

+--------------------+

| Senior Engineer |

| Staff |

| Engineer |

| Senior Staff |

| Assistant Engineer |

| Technique Leader |

| Manager |

+--------------------+

只有7種。在這種成為“坑”的列值比較少的情況下,可以考慮用“IN”來填補這個“坑”從而形成最左前綴:

EXPLAIN SELECT * FROMemployees.titlesWHERE 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 | Using where |

+----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+

這次key_len為59,說明索引被用全了,但是從type和rows看出IN實際上執行了一個range查詢,這里檢查了7個key。看下兩種查詢的性能比較:

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 ... |

+----------+------------+-------------------------------------------------------------------------------+

“填坑”后性能提升了一點。如果經過emp_no篩選后余下很多數據,則后者性能優勢會更加明顯。當然,如果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 | Using where |

+----+-------------+--------+------+---------------+------+---------+------+--------+-------------+

由于不是最左前綴,索引這樣的查詢顯然用不到索引。

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

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 | Using where |

+----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+

此時可以用到索引,但是如果通配符不是只出現在末尾,則無法使用索引。(原文表述有誤,如果通配符%不出現在開頭,則可以用到索引,但根據具體情況不同可能只會用其中一個前綴)

情況六:范圍查詢。

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 | Using where |

+----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+

范圍列可以用到索引(必須是最左前綴),但是范圍列后面的列無法用到索引。同時,索引最多用于一個范圍列,因此如果查詢條件中有兩個范圍列則無法全用到索引。

EXPLAIN SELECT * FROMemployees.titlesWHERE 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 | Using where |

+----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+

可以看到索引對第二個范圍索引無能為力。這里特別要說明MySQL一個有意思的地方,那就是僅用explain可能無法區分范圍索引和多值匹配,因為在type中這兩者都顯示為range。同時,用了“between”并不意味著就是范圍查詢,例如下面的查詢:

EXPLAIN SELECT * FROMemployees.titlesWHERE 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 | Using where |

+----+-------------+--------+-------+---------------+---------+---------+------+------+-------------+

看起來是用了兩個范圍查詢,但作用于emp_no上的“BETWEEN”實際上相當于“IN”,也就是說emp_no實際是多值精確匹配。可以看到這個查詢用到了索引全部三個列。因此在MySQL中要謹慎地區分多值匹配和范圍匹配,否則會對MySQL的行為產生困惑。

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

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

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 | Using where |

+----+-------------+--------+------+---------------+---------+---------+-------+------+-------------+

雖然這個查詢和情況五中功能相同,但是由于使用了函數left,則無法為title列應用索引,而情況五中用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 | Using where |

+----+-------------+--------+------+---------------+------+---------+------+--------+-------------+

顯然這個查詢等價于查詢emp_no為10001的函數,但是由于查詢條件是一個表達式,MySQL無法為其使用索引。看來MySQL還沒有智能到自動優化常量表達式的程度,因此在寫查詢語句時盡量避免表達式出現在查詢中,而是先手工私下代數運算,轉換為無表達式的查詢語句。

索引選擇性與前綴索引

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

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

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

Index Selectivity = Cardinality / #T

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

SELECT count(DISTINCT(title))/count(*) AS Selectivity FROMemployees.titles;+-------------+

| Selectivity |

+-------------+

| 0.0000 |

+-------------+

title的選擇性不足0.0001(精確值為0.00001579),所以實在沒有什么必要為其單獨建索引。

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

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

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 | Using where |

+----+-------------+-----------+------+---------------+------+---------+------+--------+-------------+

如果頻繁按名字搜索員工,這樣顯然效率很低,因此我們可以考慮建索引。有兩種選擇,建或,看下兩個索引的選擇性:

SELECT count(DISTINCT(first_name))/count(*) AS Selectivity FROMemployees.employees;+-------------+

| Selectivity |

+-------------+

| 0.0042 |

+-------------+

SELECT count(DISTINCT(concat(first_name, last_name)))/count(*) AS Selectivity FROMemployees.employees;+-------------+

| Selectivity |

+-------------+

| 0.9313 |

+-------------+

顯然選擇性太低,選擇性很好,但是first_name和last_name加起來長度為30,有沒有兼顧長度和選擇性的辦法?可以考慮用first_name和last_name的前幾個字符建立索引,例如,看看其選擇性:

SELECT count(DISTINCT(concat(first_name, left(last_name, 3))))/count(*) AS Selectivity FROMemployees.employees;+-------------+

| Selectivity |

+-------------+

| 0.7879 |

+-------------+

選擇性還不錯,但離0.9313還是有點距離,那么把last_name前綴加到4:

SELECT count(DISTINCT(concat(first_name, left(last_name, 4))))/count(*) AS Selectivity FROMemployees.employees;+-------------+

| Selectivity |

+-------------+

| 0.9007 |

+-------------+

這時選擇性已經很理想了,而這個索引的長度只有18,比短了接近一半,我們把這個前綴索引 建上:

ALTER TABLEemployees.employeesADD INDEX `first_name_last_name4` (first_name, last_name(4));

此時再執行一遍按名字查詢,比較分析一下與建索引前的結果:

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(即當索引本身包含查詢所需全部數據時,不再訪問數據文件本身)。

InnoDB的主鍵選擇與插入優化

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

經常看到有帖子或博客討論主鍵選擇問題,有人建議使用業務無關的自增主鍵,有人覺得沒有必要,完全可以使用如學號或身份證號這種唯一字段作為主鍵。不論支持哪種論點,大多數論據都是業務層面的。如果從數據庫索引優化角度看,使用InnoDB引擎而不使用自增主鍵絕對是一個糟糕的主意。

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

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

圖13

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

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

圖14

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

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

后記

這篇文章斷斷續續寫了半個月,主要內容就是上面這些了。不可否認,這篇文章在一定程度上有紙上談兵之嫌,因為我本人對MySQL的使用屬于菜鳥級別,更沒有太多數據庫調優的經驗,在這里大談數據庫索引調優有點大言不慚。就當是我個人的一篇學習筆記了。

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

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

參考文獻

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

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

[3] 姜承堯 著;MySQL技術內幕-InnoDB存儲引擎;機械工業出版社,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

總結

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

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

天天操天天吃 | 精品一二三区视频 | 99国产精品久久久久老师 | 国产视频一区二区三区在线 | 久久成人免费电影 | 日韩在线一二三区 | 91麻豆精品国产91久久久使用方法 | av理论电影| 啪啪午夜免费 | 日韩精品无码一区二区三区 | 福利视频一二区 | 99精品国产aⅴ | 人人爱在线视频 | 久久久精品久久日韩一区综合 | 成人三级av | 一区二区三区免费在线播放 | 美女精品久久久 | 999久久久免费精品国产 | 九九热只有这里有精品 | 国产精品毛片久久蜜 | 国产高清中文字幕 | 国产精品aⅴ | 欧美日韩另类在线观看 | 精品国产123 | 在线观看中文av | 亚洲精品国产精品国产 | 国产亚洲免费的视频看 | av在线影片 | 91精品视频免费看 | 亚洲精品综合欧美二区变态 | av在线免费网 | 香蕉视频日本 | 我要色综合天天 | 免费在线电影网址大全 | 国产96在线视频 | 在线色资源 | 国产 亚洲 欧美 在线 | 男女全黄一级一级高潮免费看 | 欧美色噜噜噜 | 亚洲精品国产日韩 | 欧美大片大全 | 亚洲更新最快 | 国产系列 在线观看 | 99视频免费 | 中文字幕2021| 色av男人的天堂免费在线 | 波多野结衣视频一区二区 | 91免费的视频在线播放 | 久久99久久99精品免费看小说 | 免费 在线 中文 日本 | 日韩成人xxxx | 丁香花在线视频观看免费 | 国产97色 | 亚洲视频一区二区三区在线观看 | 亚洲精品久久视频 | 亚洲春色奇米影视 | 美女视频黄的免费的 | 99久久99久久免费精品蜜臀 | 99在线精品观看 | 日韩影视大全 | 可以免费观看的av片 | av黄在线播放 | 国产日韩亚洲 | 99精品视频免费全部在线 | 九九九热视频 | 久草在线免费色站 | 欧美与欧洲交xxxx免费观看 | 国产一级黄色免费看 | 亚洲精品视频在线观看免费视频 | 久久综合色天天久久综合图片 | 四虎永久免费网站 | 中午字幕在线 | 在线日韩视频 | 一区二区三区久久 | 色综合中文综合网 | 99激情网| 色婷婷中文 | 日韩特黄av| 91精彩在线视频 | 日日爱网址 | 在线涩涩| 国产三级视频在线 | 国产精品美女久久久 | 成人久久视频 | 99视频在线精品 | bbw av| 在线成人国产 | 欧美乱码精品一区 | 久久综合久色欧美综合狠狠 | 成人动漫一区二区三区 | www五月天婷婷 | 久久久久久久精 | 欧美亚洲精品在线观看 | 99性视频| 国内揄拍国内精品 | 天天摸天天舔天天操 | 久久99国产精品免费 | 中文字幕在线观看播放 | 亚洲更新最快 | 99久久精品国产一区二区三区 | 97超碰伊人| 久久久久一区 | 综合久久久久久 | 亚洲一区不卡视频 | 国产中文字幕视频在线观看 | 99精彩视频在线观看免费 | 中文字幕在线久一本久 | 在线电影 一区 | 成人精品一区二区三区中文字幕 | 91在线成人 | 99久久精品国产系列 | 中文亚洲欧美日韩 | 日韩国产精品一区 | 欧美性免费 | 中文字幕在线观看不卡 | 久草视频在 | 国产精品淫片 | 久久亚洲私人国产精品va | 免费十分钟 | 久久手机免费观看 | 中文在线中文a | 中字幕视频在线永久在线观看免费 | 国产青草视频在线观看 | 国产91精品一区二区绿帽 | 91人人射 | 国产精品日韩高清 | 国产成人亚洲在线电影 | 亚洲视频www | 久久久精品欧美一区二区免费 | www.日日日.com | 久9在线 | 欧美日韩后 | av电影免费在线播放 | 黄色一级网 | 99久久爱| 在线观看涩涩 | 五月天天天操 | av在线网站观看 | 99在线免费观看 | 精品国产精品久久一区免费式 | 99国产精品久久久久久久久久 | 国产不卡精品 | 丁香婷婷色综合亚洲电影 | 96看片| 久久久久久久久久久网 | 日本大片免费观看在线 | 久精品视频 | 99精品在线看 | 绯色av一区 | 中文字幕高清有码 | 婷婷色网址 | 中文字幕电影网 | 在线精品视频免费观看 | 正在播放 国产精品 | 免费观看视频的网站 | 亚洲 欧美 精品 | 91麻豆精品国产91久久久久 | 亚洲电影成人 | 国产一级大片免费看 | 99久久久成人国产精品 | 人人爱人人爽 | www.午夜视频 | 久久久久久久久久久久影院 | 欧美精品在线免费 | 99免费在线观看视频 | 在线观看v片 | 99精品免费网 | 97看片吧 | 天天超碰| 国产一区二区精品91 | 亚洲精品视频第一页 | 国产97色| 国产福利专区 | 香蕉视频在线视频 | 欧美久久综合 | 久久久精品电影 | 免费毛片一区二区三区久久久 | 久久久精品电影 | 久久日韩精品 | 日本黄色一级电影 | 97操碰 | 日日干天天插 | 久九视频 | 日韩理论 | 最新一区二区三区 | 婷婷激情影院 | 中文免费观看 | 免费av免费观看 | 亚洲伊人天堂 | 久久免费在线视频 | 97精品国产97久久久久久春色 | 成人禁用看黄a在线 | 在线国产视频观看 | 久久免费视频这里只有精品 | 精品久久久久久亚洲综合网站 | 亚洲黄色免费在线看 | 999电影免费在线观看 | 美女黄频在线观看 | www视频在线观看 | 成人av久久| 成人精品国产免费网站 | 久一久久| 日韩在线国产精品 | 亚洲视频999| 伊人激情网 | 麻豆观看 | 在线看成人av | 五月天综合网站 | 91人人爱 | 国产一二区免费视频 | 中文字幕国产精品 | 亚洲国产精品久久久久久 | 99亚洲精品视频 | 97视频在线免费播放 | 婷婷色在线 | 亚洲精品网址在线观看 | 黄色中文字幕 | 狠狠色丁香久久婷婷综 | 国产成人精品免费在线观看 | www.久久久久| 精品欧美一区二区三区久久久 | 中文字幕在线观看完整版 | 天天操天天色天天 | 蜜桃视频色 | 欧美一性一交一乱 | 2020天天干夜夜爽 | 麻豆免费视频观看 | 日韩大片在线免费观看 | 国产精品 欧美 日韩 | av线上免费观看 | 六月丁香婷婷在线 | 欧美在线18 | 激情五月激情综合网 | 亚洲欧美精品一区 | 国产精品第54页 | 91麻豆福利 | 97免费视频在线播放 | 一区二区三区四区五区在线视频 | 久久久天天操 | 久久手机视频 | 久久噜噜少妇网站 | 日韩一区二区在线免费观看 | 久青草国产在线 | 国产成人av一区二区三区在线观看 | 国产视频资源在线观看 | 国产精品大片免费观看 | 久久精品999 | 久久999精品 | 国产在线播放一区 | 深爱五月激情五月 | 久久国产精品99久久人人澡 | 精品视频9999| 成人av观看| 久久久国产精品亚洲一区 | 一区二区三区四区五区在线视频 | 操久在线 | 五月综合久久 | 天天综合网入口 | 亚洲黄色av一区 | 亚洲欧美国产日韩在线观看 | 中文字幕第一页在线播放 | 韩国av免费 | 国产精品美女视频网站 | 在线国产小视频 | 久久国产精品视频免费看 | 中文字幕在线观看一区 | 精品国产一区二区三区久久久久久 | 久久精品国产一区二区电影 | 国产精品福利av | 色偷偷男人的天堂av | 91九色性视频 | 亚洲另类久久 | 91精品成人久久 | 国产91精品久久久久久 | 色网站在线| 在线黄色av电影 | 精品一区二区在线免费观看 | 久久免费视频4 | 久久国产乱 | 欧美极品少妇xxxxⅹ欧美极品少妇xxxx亚洲精品 | 视频在线观看91 | 中文字幕三区 | 欧美最新大片在线看 | 一区二区欧美在线观看 | 成人 亚洲 欧美 | 国产成人av福利 | 日韩精品无码一区二区三区 | av在线一级 | 91精品久久久久久久久久入口 | 在线播放亚洲激情 | 婷婷99| 最新亚洲视频 | 粉嫩av一区二区三区四区五区 | 在线免费观看国产视频 | 久久不射电影院 | 蜜桃视频成人在线观看 | 毛片网在线播放 | 国产精品一区二区av影院萌芽 | 免费久久网站 | 久久久午夜精品理论片中文字幕 | 精品国产一区二区三区日日嗨 | 一级黄色在线免费观看 | 久久视频这里有精品 | 日韩性色 | 一区二区三区日韩视频在线观看 | 欧美精品久久久久性色 | 日韩精品一区二区三区视频播放 | 日韩久久一区 | 九九热国产视频 | 国内成人精品视频 | 草在线视频 | 天天色天天 | 激情深爱.com | 最新av中文字幕 | 一区二区毛片 | 国产成人免费观看久久久 | 国产精品热 | 奇米先锋 | 日韩在线一二三区 | 国产精品成人在线观看 | 狠狠地操 | 一区二区三区在线观看免费 | www.成人久久 | 99视频99| 九九九电影免费看 | 免费又黄又爽 | 激情av网| 99爱爱| 狠狠躁夜夜躁人人爽超碰91 | 国产精品成人免费一区久久羞羞 | 在线观看理论 | 久久99国产精品久久 | 精品一区二区日韩 | 中文字幕一区在线 | 福利网址在线观看 | 在线免费高清一区二区三区 | 欧美性色网站 | 欧美午夜视频在线 | 日韩va在线观看 | 成人午夜网 | 美女视频黄,久久 | 欧美极度另类 | 午夜色性片 | 久久手机免费观看 | 欧美一区,二区 | 久久免费看视频 | av网址aaa | 久久精品视频国产 | 久久免费成人 | 色婷婷综合久久久久中文字幕1 | 久久国产成人午夜av影院宅 | 国产亚洲免费观看 | 综合天天| 在线影院中文字幕 | 天天干夜夜夜操天 | 日日夜夜骑 | 天天综合视频在线观看 | 日韩综合一区二区三区 | 国产高清在线永久 | 色www精品视频在线观看 | 五月婷久 | 伊人影院99| 久久综合色天天久久综合图片 | 精品在线视频一区二区三区 | 亚洲精品av中文字幕在线在线 | 欧美精品首页 | 亚洲人人网 | 在线观看黄色av | 亚洲国产无 | 日本视频网 | 五月婷婷婷婷婷 | 99久久激情视频 | 欧美一级片在线观看视频 | 国产福利一区二区三区视频 | 美女视频黄是免费的 | 精品中文字幕在线播放 | 亚洲第一伊人 | 久久精品爱爱视频 | 国产日韩欧美自拍 | 国产成免费视频 | 手机在线视频福利 | 日韩一区二区三区观看 | 国产精品久久久影视 | av网在线观看 | 区一区二在线 | 99热官网 | 99r在线视频 | 久久国语 | 正在播放 久久 | 成年人视频免费在线 | 久草在线在线精品观看 | 欧美一区二区三区不卡 | 最新av在线网站 | 欧洲精品在线视频 | 日本一区二区三区免费观看 | 天天综合网久久综合网 | 日韩xxxbbb | 色综合在| 久久精品网址 | 欧美xxxxx在线视频 | 日韩欧美在线高清 | 国产精品a成v人在线播放 | 最新日韩中文字幕 | 久久久久9999亚洲精品 | 日韩欧美视频在线免费观看 | 色橹橹欧美在线观看视频高清 | 婷婷视频导航 | 成人网在线免费视频 | 黄色片软件网站 | 久草在线看片 | 99re久久精品国产 | 激情久久五月 | 国产91精品在线观看 | 亚洲理论电影网 | 日韩专区在线播放 | 天天操天天干天天综合网 | 人人干天天射 | 久久伊人八月婷婷综合激情 | 国产色网| 国产精品中文字幕av | 超碰人人在 | 亚洲一二三在线 | 国产色 在线 | 五月天婷亚洲天综合网鲁鲁鲁 | 亚洲性少妇性猛交wwww乱大交 | 国产精品18久久久 | 日韩中文字幕免费视频 | 91精品秘密在线观看 | 日韩天天干 | 二区三区在线观看 | 在线免费观看视频 | 黄色av高清| 国产婷婷 | 久久精品一区八戒影视 | 国产精品久久久久久欧美 | 亚洲高清av | 久久久精品视频网站 | 六月丁香婷 | 久久躁日日躁aaaaxxxx | 国产伦精品一区二区三区高清 | www.国产高清 | 国产黄色大全 | 韩国精品一区二区三区六区色诱 | 欧美精品一区二区在线播放 | 香蕉影院在线播放 | 在线免费观看视频一区二区三区 | 久久中文网 | 成人9ⅰ免费影视网站 | 99精品视频在线播放免费 | 亚州成人av在线 | 欧美婷婷综合 | 91在线视频观看免费 | 香蕉网在线观看 | 8x成人在线 | 丁香午夜| 久久99热精品 | 久草在线视频免费资源观看 | 999成人国产 | 91av在线免费| 国产一二三在线视频 | 九九热在线播放 | 国产一二区免费视频 | 成人性生交大片免费观看网站 | 久久人人看 | 国产精品破处视频 | 国产精品免费成人 | 欧美国产日韩一区二区三区 | 欧美精品在线观看一区 | 婷婷视频导航 | 亚洲在线免费视频 | 久久人人97超碰com | 爱爱一区| 波多野结衣视频一区二区三区 | 日韩精品视频免费在线观看 | 美女视频免费精品 | 99热在线这里只有精品 | 久久成人欧美 | 久久免费中文视频 | zzijzzij日本成熟少妇 | 欧美色一色 | 狠狠色伊人亚洲综合成人 | 日日干,天天干 | 久久理伦片 | 国产粉嫩在线 | 久久国产一区 | 色偷偷av男人天堂 | 欧美成人黄色片 | 色综合久久88色综合天天人守婷 | 在线精品亚洲 | 在线观看视频一区二区三区 | 免费看国产黄色 | 在线观看www视频 | 成人av资源 | 日韩黄色免费电影 | 久青草视频在线观看 | 九九视频热 | 在线观看v片 | 狠狠色综合欧美激情 | 国产亚洲在线观看 | 人人爽人人看 | 九色在线视频 | 综合在线色 | 欧美激情视频一区 | 一区二区久久 | av福利在线 | 麻豆网站免费观看 | 成人三级黄色 | 欧美日韩国产色综合一二三四 | 久福利 | 日产乱码一二三区别在线 | 国产91国语对白在线 | 日韩欧三级 | 久久人人插| 国产成人精品综合久久久久99 | 天天天干夜夜夜操 | 91天堂素人约啪 | 激情欧美日韩一区二区 | 在线亚洲成人 | 国产黄在线 | 欧美成人理伦片 | 免费看片成年人 | 欧美专区国产专区 | 久久国产精品免费一区二区三区 | 日本中文乱码卡一卡二新区 | 日韩免费在线一区 | 亚洲国产成人精品在线观看 | 天天爽夜夜爽精品视频婷婷 | 六月激情久久 | 九九电影在线 | 久久天天躁狠狠躁亚洲综合公司 | 极品美女被弄高潮视频网站 | 久久爱www. | 青青射 | 国产精品刺激对白麻豆99 | 蜜桃视频日本 | 精品视频999 | 欧美日韩免费一区二区三区 | 欧美 高跟鞋交 xxxxhd | 激情五月***国产精品 | 成人免费观看视频网站 | 精品自拍网 | 五月婷婷中文网 | 亚洲视频精品在线 | 天天综合天天做 | 天天视频亚洲 | 女人18片毛片90分钟 | 国产精品美女久久久久久久 | 91九色网站 | 久久精品导航 | 97综合在线 | 国产精品去看片 | 成人av网站在线 | 精品二区久久 | 一级性视频 | 久久久久国产精品一区 | 日女人免费视频 | 九九九九热精品免费视频点播观看 | 91福利视频久久久久 | 亚洲成人免费 | 天海冀一区二区三区 | 国产一区在线免费观看 | 91精品国产一区二区在线观看 | 久久高清免费观看 | 国产一区免费在线 | 玖玖视频国产 | 久久中文字幕视频 | 黄色视屏免费在线观看 | 日韩视频在线不卡 | 亚洲精品视频在线观看网站 | 日韩av在线一区二区 | 亚州天堂 | 免费观看一级视频 | a黄色大片 | 在线播放国产一区二区三区 | 国产成人精品不卡 | 最近中文字幕免费 | 中文字幕免费播放 | 99在线观看视频网站 | 国产精品久久久久久久久搜平片 | av在线免费观看不卡 | 狠狠久久| 午夜久久久久久久久久影院 | 天海翼一区二区三区免费 | 成人av在线影院 | 国产精品九九热 | 亚洲精品国产精品国自产观看浪潮 | 久久国产热 | 四虎永久国产精品 | 中日韩在线 | a级免费观看| 亚洲乱码精品久久久 | 成人精品久久 | 日本不卡一区二区三区在线观看 | 中文字幕 在线看 | 五月婷在线观看 | 久久久久国产一区二区三区四区 | 国产免费大片 | 在线观看免费av网站 | 91亚洲网| 午夜久久福利视频 | a午夜电影 | 欧美另类v | 五月激情姐姐 | 黄色中文字幕在线 | 国产精品免费久久久久久久久久中文 | 成人在线视频在线观看 | 丁香花五月 | 精品福利国产 | 久久久www成人免费精品张筱雨 | 亚洲伦理中文字幕 | 亚洲麻豆精品 | 天天摸天天干天天操天天射 | 日韩精品免费专区 | 国产又粗又长的视频 | 免费在线观看一区 | 色综合天天色综合 | 婷婷电影在线观看 | 欧美综合在线视频 | 欧美乱大交 | 久久久久久久久久伊人 | 午夜视频导航 | 奇米777777 | 91视频电影 | 久久精品在线 | 黄色av成人在线 | 国产一二三区在线观看 | 免费中午字幕无吗 | 日韩精品五月天 | 久久精品在线 | a级片在线播放 | 在线中文字幕视频 | 一二区av| 午夜色站 | 亚洲黄色小说网 | 亚洲精品视频在线播放 | 美女久久视频 | 99久久99久久精品国产片果冰 | 久草在线免费资源站 | 日韩中文字幕视频在线观看 | 亚洲国产精品成人综合 | 西西444www大胆高清视频 | 成年人免费av网站 | 亚洲一本视频 | 精品国产乱码久久 | 天天操天天怕 | 国产护士在线 | 亚洲九九九在线观看 | 欧美一区二区三区特黄 | 天堂在线免费视频 | 夜夜操网 | 在线中文字幕播放 | 中文字幕丝袜一区二区 | 五月天综合色激情 | 欧美在线视频一区二区 | 国产成人一区二区三区影院在线 | 天天草天天爽 | 91精品在线免费视频 | 五月婷婷开心 | 国产一区二区三区在线 | 日韩欧美综合 | 国产乱码精品一区二区蜜臀 | 亚洲精品一区二区三区四区高清 | 日韩精品视频在线观看网址 | 欧美日韩观看 | 免费一级特黄录像 | 日韩高清免费观看 | 久草在线视频中文 | 国产精品久久久久久高潮 | 欧美成人一区二区 | 国产精品国内免费一区二区三区 | 国产精品欧美激情在线观看 | 狠狠搞,com| www.黄色片.com| 91av色| 99久久久久| 亚洲无人区小视频 | 三级午夜片 | 成人黄色毛片视频 | 欧美日韩国产成人 | 亚洲人成影院在线 | 久久久久久草 | 日日综合 | 天天射射天天 | 五月婷婷综 | 久久艹国产视频 | 成人免费 在线播放 | 日韩av偷拍 | 日韩欧美在线中文字幕 | 香蕉精品在线观看 | 日韩成人免费电影 | 波多野结衣在线播放一区 | 亚洲最新av在线 | 国产视频 久久久 | 国产精品青草综合久久久久99 | 欧美视频一区二 | 天天操欧美 | 免费大片av| 在线看av网址 | 久久亚洲综合国产精品99麻豆的功能介绍 | 国产小视频国产精品 | 草 免费视频 | 日韩激情第一页 | 精品1区二区 | 亚洲视频免费在线观看 | 99热精品视| 天天色天天操天天爽 | 亚洲日本一区二区在线 | 日韩欧美一区二区三区黑寡妇 | 久久草在线视频国产 | 国产高清在线a视频大全 | 99久久99久久精品免费 | 久久久精品99 | a级国产乱理论片在线观看 特级毛片在线观看 | 国产美女无遮挡永久免费 | 成 人 黄 色视频免费播放 | 国产精品久久久久久高潮 | 美女黄频在线观看 | 日韩av进入 | 一区二区三区高清不卡 | 亚洲精品久久久久久久不卡四虎 | 人人爱人人爽 | 中文电影网| 亚洲精品视频在 | 日本黄色一级电影 | 在线视频日韩 | 日日夜夜国产 | 国产亚洲高清视频 | 99国产精品久久久久久久久久 | 中文字幕综合在线 | 天天操天天是 | 国产亚洲婷婷 | 精品国产成人在线影院 | 久久不卡国产精品一区二区 | 亚洲好视频 | 在线看一区 | 国产a级片免费观看 | 亚洲欧洲一区二区在线观看 | 亚洲精品在线二区 | 日韩精品视频免费专区在线播放 | 丝袜网站在线观看 | 婷婷丁香国产 | 在线91色 | 久草在线一免费新视频 | 91高清完整版在线观看 | 国产传媒中文字幕 | 久久精品视频在线观看免费 | 草久在线视频 | 亚洲成人av电影在线 | 久久国产精品免费看 | 黄色官网在线观看 | 亚洲日本色 | 国产精品第7页 | 国产精品欧美久久久久无广告 | 国产精久久久久久妇女av | 欧美一二三视频 | 天天干天天爽 | 草久在线 | 久久伊人八月婷婷综合激情 | 九九热.com| 国产亚洲小视频 | av网站免费看 | 中文字幕 国产视频 | 天天射天天操天天 | 欧美激情第一页xxx 午夜性福利 | 国产亚洲精品久久久久久移动网络 | 天天艹天天 | 欧美电影在线观看 | 9999精品视频 | 亚州精品成人 | 伊人狠狠色丁香婷婷综合 | 亚洲精品在线观看av | 96久久| 久久精品综合视频 | 狠狠操在线 | 懂色av懂色av粉嫩av分享吧 | 99久久精品国产一区 | 久久99精品国产一区二区三区 | 国产精品久久久久久久久久直播 | 波多野结衣电影一区二区 | 精品久久久久_ | 国产.精品.日韩.另类.中文.在线.播放 | 国产视频观看 | 久久综合久久综合九色 | 四季av综合网站 | 国产精品一区二区麻豆 | 国产精品永久在线 | 国产精品免费在线播放 | 午夜影视一区 | 永久免费精品视频 | 91中文在线视频 | 国产精品一区二区麻豆 | 99riav1国产精品视频 | 免费看一及片 | 射久久久 | 国产视频99 | 97视频免费在线 | 日韩午夜一级片 | 欧美少妇xx| 亚洲特级毛片 | 888av| 日本性生活一级片 | 亚洲精品www久久久 www国产精品com | 欧美日韩亚洲在线观看 | 国产精品麻豆果冻传媒在线播放 | 亚洲开心激情 | 天天射综合 | 国产探花在线看 | 西西44人体做爰大胆视频 | 欧美成人黄色 | 黄色软件在线看 | 国产h在线播放 | 97精品国产一二三产区 | 狠狠操综合 | 中文字幕国产在线 | 国产综合在线观看视频 | 国产视频不卡一区 | 69国产盗摄一区二区三区五区 | 国产美女视频免费观看的网站 | 久久伊人八月婷婷综合激情 | 91mv.cool在线观看 | av在线日韩 | 夜夜操网 | 国产精品久久婷婷六月丁香 | 又爽又黄又刺激的视频 | 亚洲在线视频观看 | 五月天婷亚洲天综合网鲁鲁鲁 | 亚洲国产欧美一区二区三区丁香婷 | 免费看的黄色小视频 | 成 人 黄 色 免费播放 | 免费高清无人区完整版 | 一本一道久久a久久综合蜜桃 | 99精品国产成人一区二区 | 亚洲精品永久免费视频 | 成人中文字幕+乱码+中文字幕 | 在线观看亚洲精品 | 中文字幕久久久精品 | 狠狠狠色 | 亚洲成人黄色在线观看 | 国产精品都在这里 | 久草免费看 | 日韩精品视频在线观看免费 | a在线观看免费视频 | 免费成人在线观看 | 一区二区高清在线 | 高清久久久 | 国产在线播放一区二区三区 | 18av在线视频| 黄色av电影在线观看 | 中文字幕精品一区久久久久 | 黄色视屏免费在线观看 | 91超级碰| 日韩欧美一区二区不卡 | 午夜在线免费观看 | 亚洲激情 在线 | 欧美精品免费视频 | 国内毛片毛片 | 色欲综合视频天天天 | 成人毛片一区 | 国产精品欧美久久久久三级 | 午夜视频一区二区三区 | 国产精品美女久久久久久久 | 一区二区不卡高清 | 欧洲成人免费 | 91av视屏| 免费网站黄色 | 天天操天天射天天爱 | 欧美黑人xxxx猛性大交 | 欧美极品少妇xxxxⅹ欧美极品少妇xxxx亚洲精品 | 永久免费av在线播放 | 亚洲欧洲精品在线 | 久久精品影片 | 国产激情小视频在线观看 | 18性欧美xxxⅹ性满足 | 日韩激情综合 | 六月丁香婷婷在线 | 日韩系列 | 成年人视频在线免费观看 | 色天天| 国产福利精品在线观看 | 国产精品久久久久久影院 | 视频在线在亚洲 | 五月婷婷开心 | 国产精品激情 | 激情av一区二区 | 国产精品少妇 | 伊人天堂久久 | 国产91电影在线观看 | 久久er99热精品一区二区三区 | 不卡视频一区二区三区 | 日韩在线一级 | 天天操天天爱天天干 | 欧美一区二区免费在线观看 | 亚洲美女久久 | 国产在线精品福利 | 天天干天天草天天爽 | 麻豆视频在线 | 激情欧美一区二区三区免费看 | 四虎影视精品永久在线观看 | 亚洲精品字幕在线观看 | 亚洲免费观看在线视频 | 丁香婷婷成人 | 高清日韩一区二区 | 亚洲天堂视频在线 | 国内精品一区二区 | 91在线蜜桃臀 | 在线不卡中文字幕播放 | 最近中文字幕第一页 | 国产精品久久久久一区二区三区共 | 国产91大片 | 精品久久九九 | 久久综合久久综合久久 | 91精品老司机久久一区啪 | 久久久96 | 成人欧美日韩国产 | 97视频免费在线 | 国产乱对白刺激视频在线观看女王 | 国内丰满少妇猛烈精品播放 | 日韩欧美在线高清 | 免费看黄在线观看 | 精品999在线 | 久色婷婷 | 精品视频区 | www.五月天 | 欧美日韩高清免费 | 九九九热精品免费视频观看网站 | 亚洲欧美在线视频免费 | 十八岁以下禁止观看的1000个网站 | 久草在线视频在线观看 | 香蕉视频一级 | 亚洲天堂香蕉 | 欧美整片sss| 日韩一级理论片 | 五月av在线 | 亚洲欧美国产精品18p | 久久色在线播放 | 就色干综合 | 欧美日韩免费看 | 久久在线观看 | 免费黄色在线播放 | 一级片黄色片网站 | 国产黄免费 | 成人一级电影在线观看 | 日本三级全黄少妇三2023 | 国产精品毛片久久久久久 | 日韩在线一区二区免费 | 日韩一区二区三区视频在线 | 制服丝袜在线91 | 丁香婷婷色综合亚洲电影 | 人人爱人人舔 | 天天色天天干天天色 | av九九九| 欧美激情第一区 | 久久不见久久见免费影院 | 国产精品一区二区三区在线看 | www黄色av| 中文字幕 在线看 | 亚洲视频2 | 婷婷色综合色 | 999精品 | 激情视频在线高清看 | 国产三级精品在线 | 久久国产日韩 | 999在线视频| 欧美日韩视频免费 | 亚洲精品xx | 日韩在线观看精品 | 日韩久久久久久久久久久久 | 69国产盗摄一区二区三区五区 | 日日操操| 日本久久免费电影 | 91九色视频在线 | 中文字幕亚洲精品在线观看 | 色欧美视频 | 中文在线中文a | 2019免费中文字幕 | 日韩二区在线播放 | 免费三级骚 | 九九综合九九 | 91.精品高清在线观看 | 国产视频久久 | 人人草人 | 日韩av资源站 | 日韩一区二区三区高清免费看看 | 国产免费视频一区二区裸体 | 精品伦理一区二区三区 | 狠狠色丁香婷婷综合 | 国产资源精品 | 国精产品一二三线999 | 在线一区观看 | 综合久久精品 | 在线成人一区二区 | 天天舔天天搞 | 久久久久成人精品免费播放动漫 | 亚洲专区在线播放 | 精品久久综合 | 日韩成人精品在线观看 | 亚洲aⅴ在线观看 | 欧洲一区二区在线观看 |