天天看點

資料庫索引--SQL語句執行效率、分析及優化

資料庫表中字段上建立的索引,在寫SQL語句進行查詢時,索引可能會被使用到,也肯能因為SQL語句導緻索引在此條SQL語句執行時索引無效。

聚簇索引的順序就是資料的實體存儲順序,而對非聚簇索引的解釋是:索引順序與資料實體排列順序無關。正式因為如此,是以一個表最多隻能有一個聚簇索引。

1.關于SQL查詢效率,100w資料,查詢隻要1秒,與您分享:

機器情況

p4: 2.4

記憶體: 1 G

os: windows 2003

資料庫: ms sql server 2000

目的: 查詢性能測試,比較兩種查詢的性能

SQL查詢效率 step by step

-- setp 1.

-- 建表

create table t_userinfo

(

userid int identity(1,1) primary key nonclustered,

nick varchar(50) not null default '',

classid int not null default 0,

writetime datetime not null default getdate()

)

go

-- 建索引

create clustered index ix_userinfo_classid on t_userinfo(classid)

go

-- step 2.

declare @i int 

declare @k int

declare @nick varchar(10)

set @i = 1

while @i<1000000

begin

set @k = @i % 10

set @nick = convert(varchar,@i)

insert into t_userinfo(nick,classid,writetime) values(@nick,@k,getdate())

set @i = @i + 1

end

-- 耗時 08:27 ,需要耐心等待

-- step 3.

select top 20 userid,nick,classid,writetime from t_userinfo 

where userid not in

(

select top 900000 userid from t_userinfo order by userid asc

)

-- 耗時 8 秒 ,夠長的

-- step 4.

select a.userid,b.nick,b.classid,b.writetime from

(

select top 20 a.userid from 

(

select top 900020 userid from t_userinfo order by userid asc

) a order by a.userid desc

) a inner join t_userinfo b on a.userid = b.userid 

order by a.userid asc

-- 耗時 1 秒,太快了吧,不可以思議

-- step 5 where 查詢

select top 20 userid,nick,classid,writetime from t_userinfo 

where classid = 1 and userid not in

(

select top 90000 userid from t_userinfo 

where classid = 1

order by userid asc

)

-- 耗時 2 秒

-- step 6 where 查詢

select a.userid,b.nick,b.classid,b.writetime from

(

select top 20 a.userid from 

(

select top 90000 userid from t_userinfo

where classid = 1

order by userid asc

) a order by a.userid desc

) a inner join t_userinfo b on a.userid = b.userid 

order by a.userid asc

-- 查詢分析器顯示不到 1 秒.

查詢效率分析:

子查詢為確定消除重複值,必須為外部查詢的每個結果都處理嵌套查詢。在這種情況下可以考慮用聯接查詢來取代。

如果要用子查詢,那就用EXISTS替代IN、用NOT EXISTS替代NOT IN。因為EXISTS引入的子查詢隻是測試是否存在符合子查詢中指定條件的行,效率較高。無論在哪種情況下,NOT IN都是最低效的。因為它對子查詢中的表執行了一個全表周遊。

建立合理的索引,避免掃描多餘資料,避免表掃描!

幾百萬條資料,照樣幾十毫秒完成查詢.

二. SQL提高查詢效率

1.對查詢進行優化,應盡量避免全表掃描,首先應考慮在 where 及 order by 涉及的列上建立索引。

2.應盡量避免在 where 子句中對字段進行 null 值判斷,否則将導緻引擎放棄使用索引而進行全表掃描,如:

select id from t where num is null 

可以在num上設定預設值0,確定表中num列沒有null值,然後這樣查詢: 

select id from t where num=0

3.應盡量避免在 where 子句中使用!=或<>操作符,否則将引擎放棄使用索引而進行全表掃描。

4.應盡量避免在 where 子句中使用 or 來連接配接條件,否則将導緻引擎放棄使用索引而進行全表掃描,如: 

select id from t where num=10 or num=20 

可以這樣查詢: 

select id from t where num=10 

union all 

select id from t where num=20

5.in 和 not in 也要慎用,否則會導緻全表掃描,如: 

select id from t where num in(1,2,3) 

對于連續的數值,能用 between 就不要用 in 了: 

select id from t where num between 1 and 3

6.下面的查詢也将導緻全表掃描: 

select id from t where name like '%abc%' 

若要提高效率,可以考慮全文檢索。

7.如果在 where 子句中使用參數,也會導緻全表掃描。因為SQL隻有在運作時才會解析局部變量,但優化程式不能将通路計劃的選擇推遲到運作時;它必須在編譯時進行選擇。然而,如果在編譯時建立通路計劃,變量的值還是未知的,因而無法作為索引選擇的輸入項。如下面語句将進行全表掃描: 

select id from t where [email protected] 

可以改為強制查詢使用索引: 

select id from t with(index(索引名)) where [email protected]

8.應盡量避免在 where 子句中對字段進行表達式操作,這将導緻引擎放棄使用索引而進行全表掃描。如: 

select id from t where num/2=100 

應改為: 

select id from t where num=100*2

9.應盡量避免在where子句中對字段進行函數操作,這将導緻引擎放棄使用索引而進行全表掃描。如: 

select id from t where substring(name,1,3)='abc'--name以abc開頭的id 

select id from t where datediff(day,createdate,'2005-11-30')=0--‘2005-11-30’生成的id 

應改為: 

select id from t where name like 'abc%' 

select id from t where createdate>='2005-11-30' and createdate<'2005-12-1'

10.不要在 where 子句中的“=”左邊進行函數、算術運算或其他表達式運算,否則系統将可能無法正确使用索引。

11.在使用索引字段作為條件時,如果該索引是複合索引,那麼必須使用到該索引中的第一個字段作為條件時才能保證系統使用該索引,否則該索引将不會被使用,并且應盡可能的讓字段順序與索引順序相一緻。

12.不要寫一些沒有意義的查詢,如需要生成一個空表結構: 

select col1,col2 into #t from t where 1=0 

這類代碼不會傳回任何結果集,但是會消耗系統資源的,應改成這樣: 

create table #t(...)

13.很多時候用 exists 代替 in 是一個好的選擇: 

select num from a where num in(select num from b) 

用下面的語句替換: 

select num from a where exists(select 1 from b where num=a.num)

14.并不是所有索引對查詢都有效,SQL是根據表中資料來進行查詢優化的,當索引列有大量資料重複時,SQL查詢可能不會去利用索引,如一表中有字段sex,male、female幾乎各一半,那麼即使在sex上建了索引也對查詢效率起不了作用。

15.索引并不是越多越好,索引固然可以提高相應的 select 的效率,但同時也降低了 insert 及 update 的效率,因為 insert 或 update 時有可能會重建索引,是以怎樣建索引需要慎重考慮,視具體情況而定。一個表的索引數最好不要超過6個,若太多則應考慮一些不常使用到的列上建的索引是否有必要。

16.應盡可能的避免更新 clustered 索引資料列,因為 clustered 索引資料列的順序就是表記錄的實體存儲順序,一旦該列值改變将導緻整個表記錄的順序的調整,會耗費相當大的資源。若應用系統需要頻繁更新 clustered 索引資料列,那麼需要考慮是否應将該索引建為 clustered 索引。

17.盡量使用數字型字段,若隻含數值資訊的字段盡量不要設計為字元型,這會降低查詢和連接配接的性能,并會增加存儲開銷。這是因為引擎在處理查詢和連接配接時會逐個比較字元串中每一個字元,而對于數字型而言隻需要比較一次就夠了。

18.盡可能的使用 varchar/nvarchar 代替 char/nchar ,因為首先變長字段存儲空間小,可以節省存儲空間,其次對于查詢來說,在一個相對較小的字段内搜尋效率顯然要高些。

19.任何地方都不要使用 select * from t ,用具體的字段清單代替“*”,不要傳回用不到的任何字段。

20.盡量使用表變量來代替臨時表。如果表變量包含大量資料,請注意索引非常有限(隻有主鍵索引)。

21.避免頻繁建立和删除臨時表,以減少系統表資源的消耗。

22.臨時表并不是不可使用,适當地使用它們可以使某些例程更有效,例如,當需要重複引用大型表或常用表中的某個資料集時。但是,對于一次性事件,最好使用導出表。

23.在建立臨時表時,如果一次性插入資料量很大,那麼可以使用 select into 代替 create table,避免造成大量 log ,以提高速度;如果資料量不大,為了緩和系統表的資源,應先create table,然後insert。

24.如果使用到了臨時表,在存儲過程的最後務必将所有的臨時表顯式删除,先 truncate table ,然後 drop table ,這樣可以避免系統表的較長時間鎖定。

25.盡量避免使用遊标,因為遊标的效率較差,如果遊标操作的資料超過1萬行,那麼就應該考慮改寫。

26.使用基于遊标的方法或臨時表方法之前,應先尋找基于集的解決方案來解決問題,基于集的方法通常更有效。

27.與臨時表一樣,遊标并不是不可使用。對小型資料集使用 FAST_FORWARD 遊标通常要優于其他逐行處理方法,尤其是在必須引用幾個表才能獲得所需的資料時。在結果集中包括“合計”的例程通常要比使用遊标執行的速度快。如果開發時間允許,基于遊标的方法和基于集的方法都可以嘗試一下,看哪一種方法的效果更好。

28.在所有的存儲過程和觸發器的開始處設定 SET NOCOUNT ON ,在結束時設定 SET NOCOUNT OFF 。無需在執行存儲過程和觸發器的每個語句後向用戶端發送 DONE_IN_PROC 消息。

29.盡量避免大事務操作,提高系統并發能力。

30.盡量避免向用戶端傳回大資料量,若資料量過大,應該考慮相應需求是否合理

1、避免将字段設為“允許為空”

2、資料表設計要規範

3、深入分析資料操作所要對資料庫進行的操作

4、盡量不要使用臨時表

5、多多使用事務

6、盡量不要使用遊标

7、避免死鎖

8、要注意讀寫鎖的使用

9、不要打開大的資料集

10、不要使用伺服器端遊标

11、在程式編碼時使用大資料量的資料庫

12、不要給“性别”列建立索引

13、注意逾時問題

14、不要使用Select *

15、在細節表中插入紀錄時,不要在主表執行Select MAX(ID)

16、盡量不要使用TEXT資料類型

17、使用參數查詢

18、不要使用Insert導入大批的資料

19、學會分析查詢

20、使用參照完整性

21、用INNER JOIN 和LEFT JOIN代替Where 

///

http://blog.sina.com.cn/s/blog_4b3d79a9010006gv.html

提高SQL查詢效率(要點與技巧):

技巧一:

問題類型:ACCESS資料庫字段中含有日文片假名或其它不明字元時查詢會提示記憶體溢出。

解決方法:修改查詢語句

sql="select * from tablename where column like '%"&word&"%'"

改為

sql="select * from tablename"

rs.filter = " column like '%"&word&"%'"

===========================================================

技巧二:

問題類型:如何用簡易的辦法實作類似百度的多關鍵詞查詢(多關鍵詞用空格或其它符号間隔)。

解決方法:

'//用空格分割查詢字元串

ck=split(word," ")

'//得到分割後的數量

sck=UBound(ck)

sql="select * tablename where"

在一個字段中查詢

For i = 0 To sck

SQL = SQL & tempJoinWord & "(" & _

"column like '"&ck(i)&"%')"

tempJoinWord = " and "

Next

在二個字段中同時查詢

For i = 0 To sck

SQL = SQL & tempJoinWord & "(" & _

"column like '"&ck(i)&"%' or " & _

"column1 like '"&ck(i)&"%')"

tempJoinWord = " and "

Next

===========================================================

技巧三:大大提高查詢效率的幾種技巧

1. 盡量不要使用 or,使用or會引起全表掃描,将大大降低查詢效率。

2. 經過實踐驗證,charindex()并不比前面加%的like更能提高查詢效率,并且charindex()會使索引失去作用(指sqlserver資料庫)

3. column like '%"&word&"%' 會使索引不起作用

column like '"&word&"%' 會使索引起作用(去掉前面的%符号)

(指sqlserver資料庫)

4. '%"&word&"%' 與'"&word&"%' 在查詢時的差別:

比如你的字段内容為 一個容易受傷的女人

'%"&word&"%' :會通配所有字元串,不論查“受傷”還是查“一個”,都會顯示結果。

'"&word&"%' :隻通配前面的字元串,例如查“受傷”是沒有結果的,隻有查“一個”,才會顯示結果。

5. 字段提取要按照“需多少、提多少”的原則,避免“select *”,盡量使用“select 字段1,字段2,字段3........”。實踐證明:每少提取一個字段,資料的提取速度就會有相應的提升。提升的速度還要看您舍棄的字段的大小來判斷。

6. order by按聚集索引列排序效率最高。一個sqlserver資料表隻能建立一個聚集索引,一般預設為ID,也可以改為其它的字段。

7. 為你的表建立适當的索引,建立索引可以使你的查詢速度提高幾十幾百倍。(指sqlserver資料庫)

三.以下是建立索引與不建立索引的一個查詢效率分析:

Sqlserver索引與查詢效率分析。

表 News

字段

Id:自動編号

Title:文章标題

Author:作者

Content:内容

Star:優先級

Addtime:時間

記錄:100萬條

測試機器:P4 2.8/1G記憶體/IDE硬碟

=======================================================

方案1:

主鍵Id,預設為聚集索引,不建立其它非聚集索引

select * from News where Title like '%"&word&"%' or Author like '%"&word&"%' order by Id desc

從字段Title和Author中模糊檢索,按Id排序

查詢時間:50秒

=======================================================

方案2:

主鍵Id,預設為聚集索引

在Title、Author、Star上建立非聚集索引

select * from News where Title like '"&word&"%' or Author like '"&word&"%' order by Id desc

從字段Title和Author中模糊檢索,按Id排序

查詢時間:2 - 2.5秒

=======================================================

方案3:

主鍵Id,預設為聚集索引

在Title、Author、Star上建立非聚集索引

select * from News where Title like '"&word&"%' or Author like '"&word&"%' order by Star desc

從字段Title和Author中模糊檢索,按Star排序

查詢時間:2 秒

=======================================================

方案4:

主鍵Id,預設為聚集索引

在Title、Author、Star上建立非聚集索引

select * from News where Title like '"&word&"%' or Author like '"&word&"%'

從字段Title和Author中模糊檢索,不排序

查詢時間:1.8 - 2 秒

=======================================================

方案5:

主鍵Id,預設為聚集索引

在Title、Author、Star上建立非聚集索引

select * from News where Title like '"&word&"%'

select * from News where Author like '"&word&"%'

從字段Title 或 Author中檢索,不排序

查詢時間:1秒

? 如何提高SQL語言的查詢效率?

問:請問我如何才能提高SQL語言的查詢效率呢?

答:這得從頭說起:

   由于SQL是面向結果而不是面向過程的查詢語言,是以一般支援SQL語言的大型關系型資料庫都使用一個基于查詢成本的優化器,為即時查詢提供一個最佳的執行政策。對于優化器,輸入是一條查詢語句,輸出是一個執行政策。

    一條SQL查詢語句可以有多種執行政策,優化器将估計出全部執行方法中所需時間最少的所謂成本最低的那一種方法。所有優化都是基于用記所使用的查詢語句中的where子句,優化器對where子句中的優化主要用搜尋參數(Serach Argument)。

    搜尋參數的核心思想就是資料庫使用表中字段的索引來查詢資料,而不必直接查詢記錄中的資料。

    帶有 =、<、<=、>、>= 等操作符的條件語句可以直接使用索引,如下列是搜尋參數:

    emp_id = "10001" 或 salary > 3000 或  a =1 and c = 7

    而下列則不是搜尋參數:

    salary = emp_salary 或 dep_id != 10 或 salary * 12 >= 3000 或 a=1 or c=7

    應當盡可能提供一些備援的搜尋參數,使優化器有更多的選擇餘地。請看以下3種方法:

    第一種方法:

    select employee.emp_name,department.dep_name from department,employee where (employee.dep_id = department.dep_id) and (department.dep_code="01") and (employee.dep_code="01");

    它的搜尋分析結果如下:

    Estimate 2 I/O operations

    Scan department using primary key

    for rows where dep_code equals "01"

    Estimate getting here 1 times

    Scan employee sequentially

    Estimate getting here 5 times

    第二種方法:

    select employee.emp_name,department.dep_name from department,employee where (employee.dep_id = department.dep_id) and (department.dep_code="01");

    它的搜尋分析結果如下:

    Estimate 2 I/O operations

    Scan department using primary key

    for rows where dep_code equals "01"

    Estimate getting here 1 times

    Scan employee sequentially

    Estimate getting here 5 times

    第一種方法與第二種運作效率相同,但第一種方法最好,因為它為優化器提供了更多的選擇機會。

    第三種方法:

    select employee.emp_name,department.dep_name from department,employee where (employee.dep_id = department.dep_id) and (employee.dep_code="01");

    這種方法最不好,因為它無法使用索引,也就是無法優化……

使用SQL語句時應注意以下幾點:

    1、避免使用不相容的資料類型。例如,Float和Integer,Char和Varchar,Binary和Long Binary不相容的。資料類型的不相容可能使優化器無法執行一些本可以進行的優化操作。例如:

    select emp_name form employee where salary > 3000;

    在此語句中若salary是Float類型的,則優化器很難對其進行優化,因為3000是個整數,我們應在程式設計時使用3000.0而不要等運作時讓DBMS進行轉化。

    2、盡量不要使用表達式,因它在編繹時是無法得到的,是以SQL隻能使用其平均密度來估計将要命中的記錄數。

    3、避免對搜尋參數使用其他的數學操作符。如:

       select emp_name from employee where salary * 12 > 3000;

       應改為:

       select emp_name from employee where salary  > 250;

    4、避免使用 != 或 <> 等這樣的操作符,因為它會使系統無法使用索引,而隻能直接搜尋表中的資料。

? ORACAL中的應用

一個1600萬資料表--短信上行表TBL_SMS_MO

結構:

CREATE TABLE TBL_SMS_MO

(

 SMS_ID NUMBER,

 MO_ID VARCHAR2(50),

 MOBILE VARCHAR2(11),

 SPNUMBER VARCHAR2(20),

 MESSAGE VARCHAR2(150),

 TRADE_CODE VARCHAR2(20),

 LINK_ID VARCHAR2(50),

 GATEWAY_ID NUMBER,

 GATEWAY_PORT NUMBER,

 MO_TIME DATE DEFAULT SYSDATE

);

CREATE INDEX IDX_MO_DATE ON TBL_SMS_MO (MO_TIME)

  PCTFREE 10

  INITRANS 2

  MAXTRANS 255

  STORAGE

  (

    INITIAL 1M

    NEXT 1M

    MINEXTENTS 1

    MAXEXTENTS UNLIMITED

    PCTINCREASE 0

  );

CREATE INDEX IDX_MO_MOBILE ON TBL_SMS_MO (MOBILE)

  PCTFREE 10

  INITRANS 2

  MAXTRANS 255

  STORAGE

  (

    INITIAL 64K

    NEXT 1M

    MINEXTENTS 1

    MAXEXTENTS UNLIMITED

    PCTINCREASE 0

  );

  問題:從表中查詢某時間段内某手機發送的短消息,如下SQL語句:

SELECT MOBILE,MESSAGE,TRADE_CODE,MO_TIME

FROM TBL_SMS_MO

WHERE MOBILE='130XXXXXXXX'

AND MO_TIME BETWEEN TO_DATE('2006-04-01','YYYY-MM-DD HH24:MI:SS') AND TO_DATE('2006-04-07','YYYY-MM-DD HH24:MI:SS')

ORDER BY MO_TIME DESC

傳回結果大約需要10分鐘,應用于網頁查詢,簡直難以忍受。

分析:

在PL/SQL Developer,點選“Explain Plan”按鈕(或F5鍵),對SQL進行分析,發現預設使用的索引是IDX_MO_DATE。問題可能出在這裡,因為相對于總數量1600萬資料來說,都mobile的資料是很少的,如果使用IDX_MO_MOBILE比較容易鎖定資料。

如下優化:

SELECT MOBILE,MESSAGE,TRADE_CODE,MO_TIME

FROM TBL_SMS_MO

WHERE MOBILE='130XXXXXXXX'

AND MO_TIME BETWEEN TO_DATE('2006-04-01','YYYY-MM-DD HH24:MI:SS') AND TO_DATE('2006-04-07','YYYY-MM-DD HH24:MI:SS')

ORDER BY MO_TIME DESC

測試:

按F8運作這個SQL,哇~... ... 2.360s,這就是差别。

用索引提高SQL Server性能

特别說明

  在微軟的SQL Server系統中通過有效的使用索引可以提高資料庫的查詢性能,但是性能的提高取決于資料庫的實作。在本文中将會告訴你如何實作索引并有效的提高資料庫的性能。 

  

  在關系型資料庫中使用索引能夠提高資料庫性能,這一點是非常明顯的。用的索引越多,從資料庫系統中得到資料的速度就越快。然而,需要注意的是,用的索引越多,向資料庫系統中插入新資料所花費的時間就越多。在本文中,你将了解到微軟的SQL Server資料庫所支援的各種不同類型的索引,在這裡你将了解到如何使用不同的方法來實作索引,通過這些不同的實作方法,你在資料庫的讀性能方面得到的遠比在資料庫的整體性能方面的損失要多得多。

  

  索引的定義

  索引是資料庫的工具,通過使用索引,在資料庫中擷取資料的時候,就可以不用掃描資料庫中的所有資料記錄,這樣能夠提高系統擷取資料的性能。使用索引可以改變資料的組織方式,使得所有的資料都是按照相似的結構來組織的,這樣就可以很容易地實作資料的檢索通路。索引是按照列來建立的,這樣就可以根據索引列中的值來幫助資料庫找到相應的資料。

  

  索引的類型

  微軟的SQL Server 支援兩種類型的索引:clustered 索引和nonclustered索引。Clustered 索引在資料表中按照實體順序存儲資料。因為在表中隻有一個實體順序,是以在每個表中隻能有一個clustered索引。在查找某個範圍内的資料時,Clustered索引是一種非常有效的索引,因為這些資料在存儲的時候已經按照實體順序排好序了。

  

  Nonclustered索引不會影響到下面的實體存儲,但是它是由資料行指針構成的。如果已經存在一個clustered索引,在nonclustered中的索引指針将包含clustered索引的位置參考。這些索引比資料更緊促,而且對這些索引的掃描速度比對實際的資料表掃描要快得多。

  

  如何實作索引

  資料庫可以自動建立某些索引。例如,微軟的SQL Server系統通過自動建立唯一索引來強制實作UNIQUE限制,這樣可以確定在資料庫中不會插入重複資料。也可以使用CREATE INDEX語句或者通過SQL Server Enterprise Manager來建立其他索引,SQL Server Enterprise Manager還有一個索引建立模闆來指導你如何建立索引。

  

  得到更好的性能

  雖然索引可以帶來性能上的優勢,但是同時也将帶來一定的代價。雖然SQL Server系統允許你在每個資料表中建立多達256個nonclustered索引,但是建議不要使用這麼多的索引。因為索引需要在記憶體和實體磁盤驅動器上使用更多的存儲空間。在執行插入聲明的過程中可能會在一定程度上導緻系統性能的下降,因為在插入資料的時候是需要根據索引的順序插入,而不是在第一個可用的位置直接插入資料,這樣一來,存在的索引越多将導緻插入或者更新聲明所需要的時間就越多。

  

  在使用SQL Server系統建立索引的時候,建議參照下面的建立準則來實作:

  

  正确的選擇資料類型

  在索引中使用某些資料類型可以提高資料庫系統的效率,例如,Int,bigint, smallint,和tinyint等這些資料類型都非常适合于用在索引中,因為他們都占用相同大小的空間并且可以很容易地實作比較操作。其他的資料類型如char和varchar的效率都非常低,因為這些資料類型都不适合于執行數學操作,并且執行比較操作的時間都比上面提到資料類型要長。

  

  確定在使用的過程中正确的利用索引值

  在執行查詢操作時,可能所使用的列隻是clustered的一部分,這時尤其要注意的是如何使用這些資料。當用這些資料列作為參數調用函數時,這些函數可能會使現有的排序優勢失效。例如,使用日期值作為索引,而為了實作比較操作,可能需要将這個日期值轉換為字元串,這樣将導緻在查詢過程中無法用到這個日期索引值。

  

  在建立多列索引時,需要注意列的順序

  資料庫将根據第一列索引的值來排列記錄,然後進一步根據第二列的值來排序,依次排序直到最後一個索引排序完畢。哪一列唯一資料值較少,哪一列就應該為第一個索引,這樣可以確定資料可以通過索引進一步交叉排序。

  

  在clustered索引中限制列的數量

  在clustered索引中用到的列越多,在nonclustered索引中包含的clustered索引參考位置就越多,需要存儲的資料也就越多。這樣将增加包含索引的資料表的大小,并且将增加基于索引的搜尋時間。

  

  避免頻繁更新clustered索引資料列

  由于nonclustered 索引依賴于clustered 索引,是以如果構成clustered 索引的資料列頻繁更新,将導緻在nonclustered中存儲的行定位器也将随之頻繁更新。對于所有與這些列相關的查詢來說,如果發生記錄被鎖定的情況時,這将可能導緻性能成本的增加。

  

  分開操作(如果可能的話)

  對于一個表來說,如果需要進行頻繁的執行插入、更新操作,同時還有大量讀操作的話,在可能的情況下嘗試将這個表分開操作。所有的插入和更新操作可以在一個沒有索引的表中操作,然後将其複制到另外一個表中,在這個表裡有大量的索引可以優化讀資料的能力。

  

  适當的重建索引

  Nonclustered索引包含clustered索引的指針,這樣一來Nonclustered索引将從屬于clustered 索引。當重建clustered索引時,首先是丢棄原來的索引,然後再使用CREATE INDEX 來建立索引,或者在使用CREATE INDEX 聲明的同時将DROP_EXISTING 子句作為重建索引的一部分。将丢棄和建立分為幾步将會導緻多次重建nonclustered 索引,而不象使用DROP_EXISTING 子句那樣,隻重建一次nonclustered 索引。

  

  明智的使用填充因子

  資料存儲在那些具有固定大小的連續記憶體頁面内。随着新的記錄行的加入,資料記憶體頁将逐漸被填滿,系統就必須執行資料頁的拆分工作,通過這個拆分工作将部分資料轉移到下一個新的頁面當中。這樣的拆分之後,将加重系統的負擔,并且會導緻存儲的資料支離破碎。填充因子可以維護資料之間的缺口,一般在建立索引的時候,該索引的填充因子就已經被設定好了。這樣一來,可以減少插入資料所引起的頁面分裂的次數。因為隻是在建立索引的時候才維護空間的大小,在增加資料或者更新資料時不會去維護空間的大小。是以,要想能夠充分的利用填充因子,就必須周期性的重建索引。由填充因子所造成的缺口将導緻讀性能的下降,因為随着資料庫的擴張,越來越多的磁盤存取工作需要讀取資料。是以,在讀的次數超過寫的次數的時候,很重要的一點是考慮使用填充因子還是使用預設方式合适。

  

  管理層的決策

  通過有效的使用索引,可以在微軟的SQL Server系統中實作很好的查詢功能,但是使用索引的效率取決于幾種不同的實作決策。在索引的性能平衡方面,要做出正确的資料庫管理決策意味着需要在良好的性能和困境中抉擇。在特定的情況下,本文給出的一些建議将有助于你做出正确的決策

來源:http://blog.csdn.net/liu_jinchun/article/details/53994602