SqlServer(索引)--創建複合索引時,複合索引列順序對查詢的性能影響[轉]
https://www.cnblogs.com/wy123/p/5604400.html
SQL Server創建複合索引時,複合索引列順序對查詢的性能影響
說說複合索引
寫索引的博客太多了,一直不想動手寫,有一下兩個原因:
一是覺得有炒剩飯的嫌疑,有兄弟曾說:索引嗎,隻要在查詢條件上建索引就行了,真的可以這麼暴力嗎?
二來覺得,索引是個非常大的話題,很難概括出所有的情況,你不整出點新意來,倒是有抄襲照搬的嫌疑
既然寫了,就寫一點稍微不一樣的東西出來,
好了,廢話打住,
/*
20160814備注:今天發現一個類似的文章:https://www.cnblogs.com/fly_zj/archive/2012/08/11/2633629.html ;
可以理解為:添加組合索引時,做相等運算字段應該放在最前麵
但這麼說也不完全準確,應該說是,將選擇性高的字段,放在最前麵,通俗說就是,將最有效的過濾條件,方式複合索引的第一位
*/
搭建測試環境:
創建一張表,模擬實際業務中的一個表,往裏麵填入數據,時間字段上,相對按照時間均勻地填充,其他字段以GUID填充
Create table BusinessInfoTable
(
BuniessCode1 varchar(50),
BuniessCode2 varchar(50),
BuniessCode3 varchar(50),
BuniessCode4 varchar(50),
BuniessStatus1 tinyint,
BuniessStatus2 tinyint,
BuniessDateTime1 Datetime,
BuniessDateTime2 Datetime,
OtherColumn1 varchar(50),
OtherColumn2 varchar(50),
OtherColumn3 varchar(50)
)
declare @i int=0
while @i<1000000
begin
insert into BusinessInfoTable
values
(
NEWID(),NEWID(),NEWID(),NEWID(),RAND()100,RAND()100,
DATEADD(MI,@i,GETDATE()),DATEADD(MI,@i,GETDATE()),NEWID(),NEWID(),NEWID()
)
set @i=@i+1
end
現在有這麼一個查詢(實際上查詢遠比這個複雜,簡化一點,不要刻意造環境)
DECLARE @startDate DATETIME;
DECLARE @endDate DATETIME;
SET @startDate = CONVERT(CHAR(10), GETDATE(), 120);
SET @endDate = DATEADD(dd, 7, @startDate);
SELECT OtherColumn2,
BuniessStatus1,
BuniessStatus2,
BuniessDateTime1,
BuniessDateTime2
FROM BusinessInfoTable
WHERE BuniessDateTime1 BETWEEN @startDate AND @endDate
AND BuniessDateTime2 BETWEEN @startDate AND @endDate
AND BuniessStatus1 = 55
AND BuniessStatus2 = 66;
說明一點:
暫時不考慮聚集索引,畢竟一個表上隻能有一個聚集索引,
別人也不是傻子,不會輕易去建聚集索引,聚集索引早被占用了
既然被占用了,原則是一般不去動別人現有的東西的,比如別人建了聚集索引,你給人家刪了,根據自己的情況建聚集索引,這不是找*麼
有經驗的你一定考慮符合索引了,同時考慮到為避免Key Lookup導致的書簽查找,我們把查詢索要的OtherColumn2列include進來
比如這樣
CREATE NONCLUSTERED INDEX IDX_1 ON BusinessInfoTable
(BuniessStatus1,BuniessStatus2,BuniessDateTime1,BuniessDateTime2)
INCLUDE(OtherColumn2)
或者這樣,隻是索引前導列順序不一樣
CREATE NONCLUSTERED INDEX IDX_2 ON BusinessInfoTable
(BuniessDateTime1,BuniessDateTime2,BuniessStatus1,BuniessStatus2)
INCLUDE(OtherColumn2)
當然可以隨意調整四個列的順序,我就不過多地做演示了,有興趣的自己試
這裏的前導列的順序並不會影響到索引的使用,查詢的時候都是非聚集索引Seek,絕對的
那麼問題來了,完全一樣的查詢條件,結果一樣,使用不同的索引,索引的區別僅僅是列順序不一樣,其代價一樣嗎,有區別嗎?
同樣查詢,使用不同索引的結果(分別是上麵的IDX_1和IDX_2):
下麵看圖說話
DECLARE @startDate DATETIME;
DECLARE @endDate DATETIME;
SET @startDate = CONVERT(CHAR(10), GETDATE(), 120);
SET @endDate = DATEADD(dd, 7, @startDate);
SELECT OtherColumn2,
BuniessStatus1,
BuniessStatus2,
BuniessDateTime1,
BuniessDateTime2
FROM BusinessInfoTable WITH(INDEX(IDX_1))
WHERE BuniessDateTime1 BETWEEN @startDate AND @endDate
AND BuniessDateTime2 BETWEEN @startDate AND @endDate
AND BuniessStatus1 = 55
AND BuniessStatus2 = 66;
SELECT OtherColumn2,
BuniessStatus1,
BuniessStatus2,
BuniessDateTime1,
BuniessDateTime2
FROM BusinessInfoTable WITH(INDEX(IDX_2))
WHERE BuniessDateTime1 BETWEEN @startDate AND @endDate
AND BuniessDateTime2 BETWEEN @startDate AND @endDate
AND BuniessStatus1 = 55
AND BuniessStatus2 = 66;
原因分析:
看來是有點差別吧,好似乎這個差別還真不小。
究竟原因在何?
索引是以平衡樹(B樹)的方式存在的,複合索引的列的順序決定了B樹的信息的存儲的順序
如果是以BuniessStatus1列為前導列,因為BuniessStatus1分布的範圍(相對)較小,
這樣在查詢的時候通過BuniessStatus1=55就可以過濾出來一個比較小的結果集,後麵依次用其他條件過濾就相對較快了
比如BuniessStatus1=55過濾出來符合條件的數據有5條,加上BuniessStatus2 BuniessDateTime1 BuniessDateTime2 這三個條件再過濾,出來一條數據。
如果BuniessDateTime1 是索引的前導列,用BuniessDateTime1 between @startDate and @endDate 過濾 ,可能會有10000條數據,
然後依次再用 BuniessDateTime2,BuniessStatus1, BuniessStatus2過濾,最後也隻有一條符合條件的數據。
差別就在於:一開始的過濾條件,決定了查詢多少page初步確定滿足條件的數據,再進一步的進行過濾
如果最開始就相對精確地確定了滿足查詢條件的數據範圍,後麵可以通過相對較小的代價來最終確認出滿足條件的數據
如果最開始相對模煳地卻確定了滿足查詢條件的數據範圍,那麼這個過程的代價就相對比較大,雖然後麵通過每一個條件的過,結果是一樣的
當然這種索引的建立跟數據分布有關,但是,這裏沒有下結論說,複合索引一定要按照什麼什麼順序來是最好的
還是那句話:具體問題具體分析,避免經驗主義,沒有一刀切的手段可以解決所有的問題。
總結:
本文通過一個簡單的例子,分析了創建符合索引時,列的順序對查詢的影響,
說明在創建索引的時候,不僅僅要考慮在哪些列上創建索引,同時也要注意到,索引列的順序,是否會對查詢產生影響。
避免一說到索引,就是“在查詢條件上建索引”的暴力做法。
最後更新:2017-11-05 15:03:36