校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁 > 知識庫 > SQL Server索引的原理深入解析

SQL Server索引的原理深入解析

熱門標簽:千呼電銷機器人價格 京華物流公司地圖標注 怎樣在地圖上標注路線圖標 優質地圖標注 智能語音外呼系統選哪家 奧威地圖標注多個地方 外呼系統電銷專用 武漢長沙外呼系統方法和技巧 百度地圖標注不同路線

前言

此文是我之前的筆記整理而來,以索引為入口進行探討相關數據庫知識(又做了修改以讓人更好消化)。SQL Server接觸不久的朋友可以只看以下藍色字體字,簡單有用節省時間;如果是數據庫基礎不錯的朋友,可以全看,歡迎探討。

索引的概念

索引的用途:我們對數據查詢及處理速度已成為衡量應用系統成敗的標準,而采用索引來加快數據處理速度通常是最普遍采用的優化方法。

索引是什么:數據庫中的索引類似于一本書的目錄,在一本書中使用目錄可以快速找到你想要的信息,而不需要讀完全書。在數據庫中,數據庫程序使用索引可以重啊到表中的數據,而不必掃描整個表。書中的目錄是一個字詞以及各字詞所在的頁碼列表,數據庫中的索引是表中的值以及各值存儲位置的列表。

索引的利弊:查詢執行的大部分開銷是I/O,使用索引提高性能的一個主要目標是避免全表掃描,因為全表掃描需要從磁盤上讀取表的每一個數據頁,如果有索引指向數據值,則查詢只需要讀少數次的磁盤就行啦。所以合理的使用索引能加速數據的查詢。但是索引并不總是提高系統的性能,帶索引的表需要在數據庫中占用更多的存儲空間,同樣用來增刪數據的命令運行時間以及維護索引所需的處理時間會更長。所以我們要合理使用索引,及時更新去除次優索引。

1.聚集索引和非聚集索引

索引分為聚集索引和非聚集索引

1.1 聚集索引

表的數據是存儲在數據頁中(數據頁的PageType標記為1),SqlServer一頁是8k,存滿一頁就開辟下一頁存儲。如果表有聚集索引,那么一筆一筆物理數據就是按聚集索引字段的大小升/降排序存儲在頁中。當對聚集索引字段更新或中間插入/刪除數據時,都會導致表數據移動(造成性能一定影響),因為它要保持升/降排序。

注意,主鍵只是默認是聚集索引,它也可以設置為非聚集索引,也可以在非主鍵字段上設置為聚集索引,全表只能有一個聚集索引。

一個優秀的聚集索引字段一般包含以下4個特性:

(A).自增長

總是在末尾增加記錄,減少分頁和索引碎片。

(B).不被更改

減少數據移動。

(C).唯一性

唯一性是任何索引最理想的特性,可以明確索引鍵值在排序中的位置。

更重要的是,索引鍵指唯一的話,它在每條記錄里才可以正確指向源數據行RID。如果聚集索引鍵值不唯一,SqlServer就需要內部生成uniquifier 列組合當作聚集鍵保證“鍵值”唯一性;如果非聚集索引鍵值不唯一,就會增加RID列(聚集索引鍵或者堆表中的行指針)保證“鍵值”唯一性。

思考(可略過):索引“鍵值”在非葉子節點也有保證唯一性,原因應該是為了明確索引記錄在非葉子節點中的位置。比如有個非聚集索引字段Name2,表中有很多Name2='a'的記錄,導致Name2='a'在非葉子節點上有多條索引記錄(節點),這時候再insert一筆Name2=‘a'的記錄時,就可以根據非葉子節點的RID和新增記錄的RID很快確定要insert到哪個索引記錄(節點)上,如果沒有非葉子節點的RID,那得遍歷到所有Name2='a'的葉子節點才能確定位置。另外,當我們select * from Table1 where Name2='a'時,返回的數據是按非聚集索引Name2和RID排序的,很好理解返回的數據就是按這邊索引存儲的順序排序的。這是這條sql查詢時有用到Name2索引的結果,如果數據庫查詢計劃因“臨界點”問題選擇直接表數據掃描,那返回的數據默認就是按表數據的順序排序的。

為了“鍵值”唯一性,對于聚集索引,uniquifier 列只在索引值重復時增加。對于非聚集索引,如果創建索引時沒定義唯一,RID會在所有記錄增加,就算索引值是唯一的;如果創建索引時定義唯一,RID只在葉子層增加,用于查找源數據行,即書簽查找操作。

(D).字段長度小

聚集索引鍵長度越小,一頁索引頁就可以容納更多索引記錄,進而減少索引B樹結構的深度。例如,一個百萬記錄的表有一個int聚集索引,可能只需要3層的B樹結構。如果把聚集索引定義在更寬的列(比如uniqueidentifier列需要16 字節),那么索引的深度會增加到4層。任何聚集索引查找需要4個I/O操作(確切的說是4個邏輯讀),原先只要3個I/O操作。
同樣,非聚集索引里會包含聚集索引鍵值,聚集索引鍵長度越小非聚集索引記錄也就越小,一頁索引頁就可以容納更多索引記錄。

1.2 非聚集索引

也是存儲在頁中(PageType標記為2的頁,叫索引頁)。比如表T建立了一個非聚集索引Index_A,那么表T有100條數據的話,那么索引Index_A也就有100條數據(準確的說是100條葉子節點數據,索引是B樹結構,如果樹的高度大于0,那么就有根節點頁或中間節點頁數據,這時索引數據就超過100條),如果表T還有非聚集索引Index_B,那么Index_B也是至少100條數據,所以索引建越多開銷越大。

更新索引字段、插入一條數據、刪除一條數據都會造成索引的維護從而造成性能的一定影響。在不同情況下,性能影響是不同的。比如當你有一個聚集索引,插入的數據又都是在末尾,這樣幾乎是不會造成數據移動,影響較小;如果插入的數據在中間位置,一般會導致數據移動,而且可能產生分頁和頁碎片,影響就會稍大一點(如果插入到的中間頁有足夠的剩余空間容納插入的數據,而且位置是在頁末,也是不會造成數據移動)

2.索引的結構

都說SqlServer的索引是B樹結構(這邊假定你對B樹結構有一定了解),那它到底長什么個模樣呢,可以用Sql語句來查看它的邏輯呈現。

新建查詢執行語法: DBCC IND(Test,OrderBo,-1) --其中Test庫的OrderBo表有1萬筆數據,有聚集索引Id主鍵字段
(不妨自己動手建個表,有聚集索引字段,插入1萬表數據,然后執行這個語法看看,會收獲很多,百聞不如一見)

執行結果:

如上圖,看到一個IndexLevel=2的索引頁2112(這邊它就是B樹的根節點,IndexLevel最大的就是根節點,往下就是子級、子子級...只有一個根頁作為B樹結構的訪問入口點),說明一定還有IndexLevel=1的索引頁和IndexLevel=0的葉子頁。由于這邊是聚集索引,因此當IndexLevel=0的葉子頁就是數據頁,存儲的是一筆一筆的物理數據。如上圖也可以看到,IndexLevel=0的行的PageType等于1,就是代表數據頁,上面1.1章節講到聚集索引時,也有提到PageType=1;而如果是非聚集索引,IndexLevel=0的葉子頁,PageType是等于 2,仍然是索引頁。

同樣,我們用Sql命令DBCC PAGE看一看

-- DBCC TRACEON(3604,-1) 
DBCC PAGE(Test,1,2112,3) 
 --根節點2112,可以查出它的兩個子節點2280和2448,然后對這兩個子節點再作DBCC PAGE查詢
DBCC PAGE(Test,1,2280,3) 
DBCC PAGE(Test,1,2448,3)


如上圖,IndexLevel=2的2112頁有兩個IndexLevel=1的子節點2280和2448,子節點下又有子節點,每個節點負責不同的索引鍵值的區間(即上圖的“Id(key)”欄位,第一行值是Null,表示最小值或倒序時的最大值)。這樣的層級關系是不是就是一棵B樹結構,其中IndexLevel其實就是B樹結構中的高度Height。

SqlServer在索引中查找某一筆記錄時,是從根節點往下找到葉子節點,因為所有數據地址都有存在葉子節點,這其實是B+樹的特點之一(B樹特點是如果查找的值在非葉子節點就找到,則就能直接返回,顯然SqlServer不是這么做,要驗證這一點你可以set statistics io on把統計開起來,然后select看下邏輯讀的次數)。

既然一定會找到葉子節點,那么索引包含列只要在葉子節點記錄就可以了,即非葉子節點沒有記錄包含列,“索引包含列”見下文第3章節。

B+樹這個特點(所有數據地址都有存在葉子節點)也利于between value1 and value2 區間查詢,只要找到value1和value2(在葉子節點),然后把中間串起來就是要的結果了。

SqlServer索引結構更像是B+樹,最終是B樹和B+樹的混合版,數據結構都是人定的,不一定就是純粹的B樹或者單純的B+樹。

3.索引包含列和書簽查找

談到索引,這邊再講一個SqlServer2005開始增加的“索引包含列”功能,很實用。

比如,在大報表查詢數據時,where條件用到索引字段Name2,但是要select的字段是Name1,這時候可以使用“索引包含列”把Name1包含在索引字段Name2中,大大提高查詢性能。

語法: Create [UNIQUE]  Nonclustered/Clustered Index IndexName On dbo.Table1(Name2) Include(Name1);

接下來分析為什么索引包含列可以大大提高性能。仍然使用DBCC PAGE命令,查看一個非聚集索引并有包含列的索引數據情況:


由上圖可知,包含列Name1也存儲在索引數據中。因此,當數據庫用索引字段Name2定位到要查找的某一行時,就可以直接把Name1的值返回了,而不用再根據RID(上圖是【HEAP RID(Key)】列)定位到數據頁中去取值,即減少了書簽查找。當查詢只返回一條數據,只有一次書簽查找時當然沒什么,如果查詢返回的數據很大,每一筆都要去數據頁找數據取出來,1000筆就是1000次書簽查找,可想而知性能消耗很大,這時候“索引包含列”價值就大大體現出來了。

關于一次書簽查找,表有聚集索引(比如Id)時就是類似執行了一次 select Name1 from Table1 where Id=1 ,利用聚集索引鍵Id查找(查找方式就是索引Id的B樹結構查找),而如果表沒有聚集索引,則是根據數據行指針(由“文件號2byte:頁號4byte:槽號2byte”組成)查找。聚集索引鍵和行指針一般統稱為RID(Row ID)指針。從這里我們可以想到,如果你的表沒有很好的聚集索引字段,建議自增長的Id字段做聚集索引主鍵(冗余出Id字段也行),它符合自增長、不被更改、唯一性、長度小的特性,是聚集索引的很好選擇。

自增長Id絕大部分情況下是適用的,特殊的情況看具體需求而定吧。還有自增長Id要考慮一個缺陷,當對表大數據量的并發insert記錄時,可以想象每個線程都是要insert到末尾那個頁,就會發生競爭和等待。解決這種情況你可以用uniqueidentifier類型字段(16字節,我是不建議使用)或者哈希分區(就是一個表分成多個表,大數據處理中分庫分表是正常的)等。但是我建議先優化你的insert效率(insert性能本身是很快的),測試每秒并發insert數是否滿足生產環境,以保留簡單穩定高效的自增長Id作法。

自增長Id不一定就是用數據庫提供的自增長,你也可以自己寫算法生成一個并發情況下也能唯一的Id(這時候一般長度是bitint,8字節整形),這種情況適合場景是分布式數據庫中主從復制時Id欄位是要求一定不能出錯的情況(主從復制的一般模式下,主庫的Id是按主庫增長,從庫Id也是按從庫自己的增長,如果遇到死鎖等原因導致主從復制不同步時,那從庫的Id就和主庫的Id自增長就對不上號了)。如果自增長Id是冗余出的主鍵,那主從庫Id對不上號也就無影響。

另外,上圖最后一列【Row Size】還告訴我們,索引列或索引包含列的size不要太長,否則一頁容不了幾筆記錄,這樣大大增加了索引頁數量,而且索引數據所占的空間也大大增加了。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

您可能感興趣的文章:
  • sqlserver索引的原理及索引建立的注意事項小結
  • SQL Server2014 哈希索引原理詳解
  • SqlServer索引的原理與應用詳解
  • SQL Server 索引介紹
  • SQLSERVER全文目錄全文索引的使用方法和區別講解
  • SQL Server 聚集索引和非聚集索引的區別分析
  • SQLSERVER 創建索引實現代碼
  • SQLSERVER聚集索引和主鍵(Primary Key)的誤區認識
  • sqlserver 索引的一些總結
  • SQL Server全文索引服務

標簽:銅仁 益陽 威海 天水 來賓 七臺河 宿州 防疫戰設

巨人網絡通訊聲明:本文標題《SQL Server索引的原理深入解析》,本文關鍵詞  SQL,Server,索引,的,原理,深入,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《SQL Server索引的原理深入解析》相關的同類信息!
  • 本頁收集關于SQL Server索引的原理深入解析的相關信息資訊供網民參考!
  • 推薦文章
    校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃
    五月天婷婷综合| 欧美性受xxxx黑人xyx| 天天综合天天综合色| 亚洲天堂成人在线观看| 国产日韩欧美精品一区| 久久久777精品电影网影网| www一区二区| 国产区在线观看成人精品| 久久精品日产第一区二区三区高清版 | 欧美日韩亚洲丝袜制服| 欧美亚洲综合一区| 欧美在线视频不卡| 日韩一区二区三区免费看| 欧美成人三级在线| 国产午夜亚洲精品不卡| 欧美国产综合一区二区| 国产精品乱子久久久久| 亚洲欧美国产三级| 视频一区在线视频| 久久99国产精品久久| 国产一区二区三区久久悠悠色av| 国产成人免费视频网站| 色爱区综合激月婷婷| 欧美一区二区精品| 国产精品久久久久久久岛一牛影视| 成人欧美一区二区三区小说| 亚洲福利视频一区| 国产自产2019最新不卡| av综合在线播放| 欧美日韩五月天| 久久综合精品国产一区二区三区| 中文字幕国产精品一区二区| 亚洲成人激情自拍| 国产精品自产自拍| 欧美在线免费播放| 久久久久亚洲蜜桃| 亚洲成国产人片在线观看| 国产永久精品大片wwwapp| 91在线云播放| 欧美电视剧免费全集观看| 一区二区三区在线高清| 激情成人综合网| 欧美午夜精品免费| 国产精品免费av| 日本不卡在线视频| av在线播放成人| 久久影院午夜片一区| 婷婷夜色潮精品综合在线| 99久久久国产精品免费蜜臀| 欧美精品一区二区三区在线 | 9久草视频在线视频精品| 欧美日本乱大交xxxxx| 国产女同性恋一区二区| 午夜影视日本亚洲欧洲精品| 99精品视频一区二区三区| 精品国内二区三区| 五月天激情综合| 972aa.com艺术欧美| 久久久午夜电影| 韩国在线一区二区| 欧美一区二区三区爱爱| 午夜激情久久久| 色999日韩国产欧美一区二区| 国产精品无遮挡| 国产黑丝在线一区二区三区| 日韩欧美在线一区二区三区| 亚洲风情在线资源站| 色婷婷久久99综合精品jk白丝 | 亚洲乱码精品一二三四区日韩在线| 韩国成人精品a∨在线观看| 正在播放一区二区| 一个色在线综合| 在线国产电影不卡| 亚洲在线视频免费观看| 色综合天天综合狠狠| 亚洲女同一区二区| 色综合一区二区三区| 日韩理论片在线| 93久久精品日日躁夜夜躁欧美| 中文字幕一区二区三区四区| www.爱久久.com| 亚洲综合清纯丝袜自拍| 欧美日韩国产大片| 免费看欧美美女黄的网站| 精品少妇一区二区三区视频免付费 | 欧美日韩三级在线| 天天色天天操综合| 欧美一二三区在线观看| 久久aⅴ国产欧美74aaa| 久久先锋资源网| youjizz久久| 午夜精品影院在线观看| 欧美一级欧美三级在线观看| 精品一区二区影视| 国产精品蜜臀av| 欧美日韩成人高清| 狠狠久久亚洲欧美| 亚洲日本韩国一区| 欧美蜜桃一区二区三区| 激情五月激情综合网| 中文字幕乱码日本亚洲一区二区 | 精品国产制服丝袜高跟| 国产成a人无v码亚洲福利| 中文字幕欧美一区| 欧美日韩一区二区三区在线| 精品一区中文字幕| 亚洲视频综合在线| 3751色影院一区二区三区| 国产精品原创巨作av| 国产精品高潮久久久久无| 678五月天丁香亚洲综合网| 美女视频黄 久久| 国产精品久久毛片| 欧美一区二区三区播放老司机| 国产酒店精品激情| 视频在线观看91| 亚洲免费伊人电影| 精品少妇一区二区三区 | 亚洲一区中文在线| 久久综合色婷婷| 欧美性生交片4| 成人aa视频在线观看| 免费一级欧美片在线观看| 亚洲欧美另类久久久精品| 精品日韩99亚洲| 欧美日韩午夜在线视频| 白白色 亚洲乱淫| 国产一区二区三区黄视频 | 精品久久人人做人人爱| 在线视频观看一区| 成人白浆超碰人人人人| 国产精品一区二区在线观看网站| 亚洲一区二区高清| 一区二区三区日本| 亚洲丝袜美腿综合| 国产精品三级在线观看| 精品91自产拍在线观看一区| 99久久精品国产一区二区三区| 国产精品久久久一区麻豆最新章节| 在线综合视频播放| 91国偷自产一区二区开放时间| 国产一区二区按摩在线观看| 午夜激情久久久| 亚洲电影视频在线| 亚洲一级不卡视频| 亚洲精品成人在线| 专区另类欧美日韩| 国产精品福利一区二区| 国产精品免费免费| 久久久国产综合精品女国产盗摄| 日韩视频一区二区三区在线播放| 91福利在线看| 色噜噜偷拍精品综合在线| 91色视频在线| 欧美日韩一区二区欧美激情 | 欧美系列在线观看| 欧美在线视频日韩| 在线视频欧美精品| 欧美老女人第四色| 欧美一区二区大片| 欧美tickling网站挠脚心| 欧美电影免费观看高清完整版| 精品三级av在线| 久久久精品2019中文字幕之3| 国产色一区二区| 亚洲日本va在线观看| 亚洲综合色网站| 蜜桃av一区二区| 欧美日本韩国一区二区三区视频| 免费久久精品视频| 亚洲国产精品影院| 丝袜美腿亚洲一区| 热久久国产精品| 国产高清不卡二三区| 欧美不卡一区二区| 成人夜色视频网站在线观看| 免费视频最近日韩| 国产成人免费高清| 91蜜桃网址入口| 欧美日韩国产综合视频在线观看| 制服.丝袜.亚洲.另类.中文| 精品福利一区二区三区免费视频| 日韩精品一区二区三区中文不卡 | 国产日韩欧美高清| 国产精品电影院| 午夜视频一区二区| 国产剧情一区二区三区| 欧美午夜电影一区| 久久亚洲精华国产精华液 | 国产婷婷色一区二区三区四区| 国产精品乱人伦中文| 日韩av不卡一区二区| 国产·精品毛片| 日韩美女一区二区三区四区| 久久久亚洲午夜电影| 一区二区三国产精华液| 久久成人免费电影| 色综合天天综合网天天狠天天| 日韩欧美电影在线| 一区二区三区中文在线观看|