通過空間占用和執行計划了解SQL Server的行存儲索引

来源:https://www.cnblogs.com/kernelry/archive/2023/05/11/17392438.html
-Advertisement-
Play Games

1 索引介紹 索引是一種幫助查詢語句能夠快速定位到數據的一種技術。索引的存儲方式有行存儲索引、列存儲索引和記憶體優化三種存儲方式: 行存儲索引,使用B+樹結構,行存儲指的是數據存儲格式為堆、聚集索引和記憶體優化表的表,用於OLTP場景。行存儲索引按順序排列的值列表,每個值都有指向其所在的數據頁面的指針。 ...


1 索引介紹

索引是一種幫助查詢語句能夠快速定位到數據的一種技術。索引的存儲方式有行存儲索引、列存儲索引和記憶體優化三種存儲方式:

  1. 行存儲索引,使用B+樹結構,行存儲指的是數據存儲格式為堆、聚集索引和記憶體優化表的表,用於OLTP場景。行存儲索引按順序排列的值列表,每個值都有指向其所在的數據頁面的指針。
    • 聚集索引
    • 非聚集索引
    • 唯一索引
    • 篩選索引
  2. 列存儲索引,使用列結構存儲,列存儲指的是在邏輯上整理為包含行和列的表,實際上以列式數據格式存儲的數據,用於OLAP場景。使用基於列的數據存儲和查詢處理。
    • 聚集列存儲
    • 非聚集列存儲
  3. 記憶體優化索引,使用Bw樹存儲,Bw樹使用一種“旋轉”技術,更適合處理處理範圍查詢和隨機插入/刪除操作,適用於各種場景下的數據存儲和查詢。
    本文中我們討論的索引就是行存儲索引中的聚集索引和非聚集索引,不涉及其它索引。

Bw樹使用一組新的旋轉技術,支持更加高效的範圍查詢操作。而B+樹則使用葉節點鏈表來處理範圍查詢。在B+樹中,如果您需要範圍查詢,您需要遍歷整個鏈表,這會增加查詢的時間成本。相比之下,Bw樹通過一些特殊的旋轉操作,能夠使得範圍查詢操作更加高效,從而顯著提高查詢性能。
假設需要查詢數字在100到200之間的數據,那麼B+樹需要遍歷相應的葉節點鏈表,而Bw樹則可以使用一些特殊的旋轉操作,跳過某些節點,快速定位到相應的數據範圍,從而減少了查詢的時間成本。
總體來說,Bw樹在範圍查詢和隨機操作等特殊情況下比B+樹更加高效。但是對於其他類型的查詢操作,它們的性能並沒有很大的區別,具體的效果需要根據應用場景來進行具體分析。

2 行存儲索引的數據組織結構

聚集索引和非聚集索引都是使用B+樹結構組織的,最頂層稱為根節點,中間層稱為中間節點,最底層稱為葉節點。在聚集索引中,葉節點包含了基礎表的數據頁,根節點和中間節點包含了索引行的索引頁,每個索引行包含一個鍵值和一個指針,通過指針來找到某個葉節點的數據行。而在非聚集索引中,葉節點只包含了索引行的索引頁,沒有數據頁,它的索引行中只有指針,通過指針來找到對應的堆表的RID或者聚集索引的數據頁。
聚集索引和非聚集索引的數據組織結構
聚集索引決定了表中數據行的存儲順序(升序/降序),所以每張表只能有1個聚集索引,可以使用CREATE CLUSTERED INDEX來手動創建聚集索引,也可以是在建表時指定主鍵的方式來自動創建。
每張表可以有多個非聚集索引,可以針對不同的查詢語句和業務場景來創建非聚集索引,只能是使用CREATE NONCLUSTERED INDEX來手動創建非聚集索引。

3 兩種索引的空間占用對比

由於聚集索引的葉節點存儲了是數據頁,由中間節點存放了指針,而非聚集索引的葉節點存放了指針(行定位器),那通過B+樹的構造,可以大概判斷是非聚集索引要消耗的空間更多,因為非聚集索引要存放更多的指針信息(葉節點的數量肯定會比中間節點的數量多)。

3.1 使用sp_spaceused查看索引大小

  1. 查看基礎表order_line,目前行數1232537行,數據大小約80MB,未創建索引。
    使用exec sp_spaceused order_line命令查看。
  2. 在order_line表的ol_w_idol_d_idol_o_idol_number列上創建聚簇索引 order_line_i1_clustered
    CREATE UNIQUE CLUSTERED INDEX [order_line_i1_clustered] ON [dbo].[order_line]
    (
    	[ol_w_id] ASC,
    	[ol_d_id] ASC,
    	[ol_o_id] ASC,
    	[ol_number] ASC
    )WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, SORT_IN_TEMPDB = OFF, IGNORE_DUP_KEY = OFF, DROP_EXISTING = OFF, ONLINE = OFF, ALLOW_ROW_LOCKS = OFF, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
    GO   
    
  3. 查看表的索引大小,約232KB,說明聚簇索引order_line_i1_clustered的大小為232KB-24KB=208KB。
    使用exec sp_spaceused order_line命令查看。
  4. 在order_line表的ol_w_id、ol_d_id、ol_o_id和ol_number列上創建非聚簇索引order_line_i1_nonclustered
    CREATE UNIQUE CLUSTERED INDEX [order_line_i1_clustered] ON [dbo].[order_line]
    (
    	[ol_w_id] ASC,
    	[ol_d_id] ASC,
    	[ol_o_id] ASC,
    	[ol_number] ASC
    )WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, SORT_IN_TEMPDB = OFF, IGNORE_DUP_KEY = OFF, DROP_EXISTING = OFF, ONLINE = OFF, ALLOW_ROW_LOCKS = OFF, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
    GO
    
  5. 查看表的索引大小,約19MB,說明非聚簇索引order_line_i1_clustered的大小為18MB~19MB。
    使用exec sp_spaceused order_line命令查看。

3.2 使用DBCC查看索引大小

我們也可以通過另外一種方式來證明,通過查詢索引ID,再使用dbcc ind將索引的所有頁返回,然後再計算索引頁的結果

  1. 首先查看兩個表的查詢索引ID
     SELECT t.name AS TableName,i.name AS IndexName,i.index_id,i.type_desc
    FROM sys.dm_db_partition_stats AS s
    INNER JOIN sys.indexes AS i 
      ON s.object_id = i.object_id 
      AND s.index_id = i.index_id
    INNER JOIN sys.tables AS t 
      ON t.object_id = i.object_id
    WHERE t.name='order_line'
    
  2. 將兩個索引的DBCC IND結果輸出到dbcc_ind_result表中,然後計算索引的大小
    CREATE TABLE dbcc_ind_result (
        PageFID int,
        PagePID int,
        IAMFID int,
        IAMPID int,
        ObjectID int,
        IndexID int,
        PartitionNumber int,
        PartitionID bigint,
        iam_chain_type varchar(30),
        PageType int,
        IndexLevel int,
        NextPageFID int,
        NextPagePID int,
        PrevPageFID int,
        PrevPagePID int
    );
    GO
    INSERT INTO dbcc_ind_result exec('DBCC IND(0,order_line,1)');
    GO
    INSERT INTO dbcc_ind_result exec('DBCC IND(0,order_line,5)');
    GO
    SELECT d.IndexID,i.name,COUNT(*)  AS PageCount,COUNT(*)*8 AS SizeKB
    FROM dbcc_ind_result d 
    INNER JOIN sys.indexes AS i 
    ON d.ObjectID = i.object_id 
    AND d.IndexID = i.index_id
    WHERE d.PageType=2 
    GROUP BY d.IndexID,i.name
    GO
    

    實驗證明,在相同的列上,非聚集索引比聚集索引需要更多的空間來存放指針信息(行定位器),消耗更多的空間。

4 兩種索引讀取數據的方式

前文提到聚集索引的葉節點存放的是數據頁,而非聚集索引葉節點存放的是指針來指向數據的位置,數據的位置可以是堆(head)的RID,也可以時聚集索引的葉節點。下麵創建一張測試表來驗證。

4.1 未創建索引時

  1. 創建測試表,生產10000行測試數據
    DROP TABLE IF EXISTS dbo.Test1;
    CREATE TABLE dbo.Test1 (
        C1 INT,
        C2 INT);
    WITH Nums
    AS (SELECT TOP (10000)
            ROW_NUMBER() OVER (ORDER BY (SELECT 1)) AS n
        FROM master.sys.all_columns AS ac1
            CROSS JOIN master.sys.all_columns AS ac2)
    INSERT INTO dbo.Test1 (
        C1,
        C2)
    SELECT n,
           2
    FROM Nums;
    
  2. 打開統計信息和執行計劃功能, 從10000行中查詢1行數據,例如查詢C1列為1000的數據。
    SET STATISTICS TIME;
    SET STATISTICS IO;
    SELECT t.C1,t.C2
    FROM dbo.Test1 AS t
    WHERE C1 = 1000;
    
    執行後可以看到統計信息項,發生了22個邏輯讀:
    • 表 'Test1'。掃描計數 1,邏輯讀取 22 次,物理讀取 0 次,預讀 0 次,lob 邏輯讀取 0 次,lob 物理讀取 0 次,lob 預讀 0 次。
    • 並且執行計劃中使用了全表掃描,需要讀取10000行數據。

4.2 創建非聚集索引後

在C1列創建1個非聚集索引後,再觀察統計信息和執行計劃是否發生變化

  1. 創建非聚集索引
    CREATE NONCLUSTERED INDEX incl ON dbo.Test1(C1);
    
    創建非聚集索引的過程中,消耗了和前一個查詢相同的資源,統計信息一樣:
    • 表 'Worktable'。掃描計數 0,邏輯讀取 0 次,物理讀取 0 次,預讀 0 次,lob 邏輯讀取 0 次,lob 物理讀取 0 次,lob 預讀 0 次。
    • 表 'Test1'。掃描計數 1,邏輯讀取 22 次,物理讀取 0 次,預讀 0 次,lob 邏輯讀取 0 次,lob 物理讀取 0 次,lob 預讀 0 次。
  2. 執行相同的查詢語句,觀察統計信息和執行計劃
    這一次統計信息發生了變化,比沒有索引的情況下消耗的邏輯讀更少,只發生了3個邏輯讀:
    • 表 'Test1'。掃描計數 1,邏輯讀取 3 次,物理讀取 0 次,預讀 0 次,lob 邏輯讀取 0 次,lob 物理讀取 0 次,lob 預讀 0 次。
    • 而執行計劃則由Table SCAN變為了Index Seek和RID,先是掃描非聚集索引中特定範圍的行,該行的指針信息為Bmk1000,再將該指針信息到堆中的RID,再返回數據,這個過程在表中只需要讀取1行數據。

4.3 創建聚集索引後

在非聚集索引的基礎上,我們再創建一個聚集索引,通過語句的執行計劃來瞭解讀取數據的方式。

  1. 創建聚集索引
    CREATE CLUSTERED INDEX icl ON dbo.Test1(C1);
    
    創建聚集索引的過程中,產生的統計信息要比非聚集要多,消耗資源也要更多:
    • 表 'Test1'。掃描計數 1,邏輯讀取 22 次,物理讀取 0 次,預讀 0 次,lob 邏輯讀取 0 次,lob 物理讀取 0 次,lob 預讀 0 次。
    • 表 'Test1'。掃描計數 1,邏輯讀取 24 次,物理讀取 0 次,預讀 0 次,lob 邏輯讀取 0 次,lob 物理讀取 0 次,lob 預讀 0 次。
      再來看看執行計劃,由於再4.2中創建了非聚集索引,執行計劃里將創建聚集索引的操作拆成了兩條語句,並且還是INSERT語句:
    • 查詢1:首先還是對錶進行了一次全表掃描,並且按照升序的方式進行了排序後,再將數據插入到聚集索引裡面。這裡對應的就是邏輯讀取22次這條統計信息,完成了整個聚集索引的創建。
    • 查詢2:然後對整個聚集索引掃描,並將非聚集索引的指針信息更新為聚集索引的葉節點。這裡對應的就是邏輯讀取24次這條統計信息,完成了整個非聚集索引的指針信息更新。
  2. 再次執行相同的查詢語句,消耗的邏輯讀比非聚集索引要少,只需要2次邏輯讀
    • 表 'Test1'。掃描計數 1,邏輯讀取 2 次,物理讀取 0 次,預讀 0 次,lob 邏輯讀取 0 次,lob 物理讀取 0 次,lob 預讀 0 次。
    • 執行計劃也不再需要使用非聚集索引和堆的RID返回數據
  3. 繼續驗證非聚集索引是否會通過聚集索引來返回數據,需要使用提示語法來固定語句使用非聚集索引。
    SELECT t.C1,t.C2
    FROM dbo.Test1 AS t WITH(INDEX = incl)
    WHERE C1 = 1000;
    
    發現這種讀取數據的方式要消耗更多的邏輯讀,比RID多了1次邏輯讀,比聚集索引多了2次邏輯讀:
    • 表 'Test1'。掃描計數 1,邏輯讀取 4 次,物理讀取 0 次,預讀 0 次,lob 邏輯讀取 0 次,lob 物理讀取 0 次,lob 預讀 0 次。
    • 執行計劃中先到非聚集索引查找C1=1000所在的行,然後再將輸出的指針信息Uniq1001到聚集索引中執行鍵值查找,返回數據。

5 行存儲索引的基礎總結

行存儲索引的聚集索引和非聚集索引在生產環境上普遍都會使用到,在本文的基礎上,我們進行簡單總結。

  1. 在數據組織結構上
    聚集索引的葉節點存儲的是數據頁,決定了表數據的排序方式;非聚集索引的葉節點存儲的是指針(行定位器),有可能是堆的RID,也有可能是聚集索引的指針。
  2. 在空間占用上
    聚集索引只需要很小的空間來存儲數據頁的信息和順序;非聚集索引需要存儲數據的指針,占用空間大。
  3. 在讀取數據的方式上
    聚集索引直接通過葉節點讀取數據頁;非聚集索引需要通過指針找到RID或者聚集索引的指針,再通過聚集索引查找鍵值。
  4. 在邏輯讀的次數上
    直接讀聚集索引,邏輯讀最小,測試邏輯讀次數為2
    通過非聚集索引+RID,邏輯讀居中,測試邏輯讀次數為3
    通過聚集索引+非聚集索引,邏輯讀最大,測試邏輯讀次數為4
  5. 在創建方式上
    聚集索引:創建主鍵時自動使用主鍵列為聚集索引,沒有主鍵時可以通過CRAETE CLUSTERED INDEX 創建,可以指定多個列;每張表只能有1個聚集索引。
    非聚集索引:手動創建,通過CRAETE NONCLUSTERED INDEX 創建;每張表可以有多個非聚集索引。

本次僅對索引的基本知識進行介紹,後續再根據不同的使用場景來驗證和說明。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.S5PV210 概述 1.1 架構概述 S5PV210 是一款 32 位 RISC 高性價比、低功耗、高性能的微處理器解決方案,適用於手機和一般應用。它集成了ARM Cortex-A8內核,實現了ARM架構V7-A,支持外設。 為3G和3.5G通信業務提供優化的硬體(H/W)性能,S5PV210 ...
  • 哈嘍大家好,我是鹹魚。今天給大家分享一個關於 HTTP 有趣的現象 鏈接:https://csvbase.com/meripaterson/stock-exchanges 我們用瀏覽器訪問這個鏈接,可以看到下麵的網頁 但如果我們使用 curl 命令去訪問這個鏈接呢? 可以看到返回的是一個 csv 文 ...
  • 1、創建文件夾 創建文件夾主要是為了後面東西打包方便。名字隨意。我這裡創建名為:Macos 2、下載印表機驅動 印表機驅動格式一般是pkg格式,其他格式的暫未測試,如果是其他格式的可以直接看我的代碼怎麼寫的參考一下我的邏輯。 3、創建配置文件 註意‼️:在剛纔創建的文件夾下午創建,這一步創建的主要用 ...
  • 目錄 一、進程原理 二、進程工作過程 三、進程類型 四、管理進程 五、前後臺調度 六、定時任務 七、管理定時任務 一、進程原理 進程:指正在運行的程式稱之為進程 程式:指的是沒有運行的代碼 線程:真正的執行者,是複製了一部分進程的代碼,可以完成部分任務的,真正的執行者 二、進程工作過程 進程工作過程 ...
  • 環境: 系統版本:Windows 10 需求描述: 描述:在搜索引擎中想在結果中過濾一些指定網站,快速找到有價值的內容。 實現方法-視頻與文字教程: 視頻教程: 文字教程: 1.進入百度搜索引擎,bing搜索引擎貌似不可以。 www.baidu.com 2.搜索內容如:銳捷無線AC基礎命令 -csd ...
  • 前言 平時大家創建項目基本上都是藉助開發工具創建,比如visual studio,visual studio code,今天我們在Linux系統上,通過命令行的形式創建.NET6項目。 版本介紹 系統版本:Ubuntu22.04 SDK版本:.NET6 安裝.NET環境 # 升級系統 sudo ap ...
  • 一、安裝 Mailx Ubuntu sudo apt-get install bsd-mailx CentOS7 yum -y install mailx 二、獲取郵箱授權碼 (解決報錯問題 535 Error: authentication failed, system busy) 以騰訊企業郵箱 ...
  • 本篇記錄在macOS系統下使用VS Code配置 GCC、GDB來搭建C/C++開發環境,首先要提前安裝好gcc和gdb,見前一篇博文 macOS下安裝gcc、gdb(親測可行) 安裝好gcc、gdb之後,接下來為VS Code配置文件使其可以調用gcc、gdb,macOS下的配置過程與Linux下 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...