欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

SQL中WHERE變量IS NULL條件導(dǎo)致全表掃描問(wèn)題的解決方法

 更新時(shí)間:2013年09月26日 15:34:33   作者:  
今天在評(píng)審接手的項(xiàng)目中的存儲(chǔ)過(guò)程時(shí),發(fā)現(xiàn)存在大量的在條件里判斷變量是否NULL的寫(xiě)法

復(fù)制代碼 代碼如下:

SET @SQL = 'SELECT * FROM Comment with(nolock) WHERE 1=1
    And (@ProjectIds Is Null or ProjectId = @ProjectIds)
    And (@Scores is null or Score =@Scores)'


印象中記得,以前在做Oracle開(kāi)發(fā)時(shí),這種寫(xiě)法是會(huì)導(dǎo)致全表掃描的,用不上索引,不知道Sql Server里是否也是一樣呢,于是做一個(gè)簡(jiǎn)單的測(cè)試
1、建立測(cè)試用的表結(jié)構(gòu)和索引:
復(fù)制代碼 代碼如下:

CREATE TABLE aaa(id int IDENTITY, NAME VARCHAR(12), age INT)
go
CREATE INDEX idx_age ON aaa (age)
GO

2、插入1萬(wàn)條測(cè)試數(shù)據(jù):

復(fù)制代碼 代碼如下:

DECLARE @i INT;
SET @i=0;
WHILE @i<10000
BEGIN
  INSERT INTO aaa (name, age)VALUES(CAST(@i AS VARCHAR), @i)
  SET @i=@i+1;
END
GO

3、先開(kāi)啟執(zhí)行計(jì)劃顯示:
在SQL Server Management Studio的查詢窗口里,右擊窗口任意位置,選擇“包含實(shí)際的執(zhí)行計(jì)劃”:

4、開(kāi)始測(cè)試,用下面的SQL進(jìn)行測(cè)試:

復(fù)制代碼 代碼如下:

DECLARE @i INT;
SET @i=100
SELECT * FROM aaa WHERE (@i IS NULL OR age = @i)
SELECT * FROM aaa WHERE (age = @i OR @i IS NULL)
SELECT * FROM aaa WHERE age=isnull(@i, age)
SELECT * FROM aaa WHERE age = @i

測(cè)試結(jié)果如下:

可以看到,即使@i有值,不管@i IS NULL是放在前面還是放在后面,都無(wú)法用到age的索引,另外age=ISNULL(@i,age)也用不上索引

最終結(jié)論,SQL Server跟ORACLE一樣,如果條件里加了 變量 IS NULL,都會(huì)導(dǎo)致全表掃描。

建議SQL改成:

復(fù)制代碼 代碼如下:

DECLARE @i INT;
SET @i=100

DECLARE @sql NVARCHAR(MAX)
SET @sql = 'SELECT * FROM aaa'
IF @i IS NOT NULL
    SET @sql = @sql + ' WHERE age = @i'
EXEC sp_executesql @sql, N'@i int', @i


當(dāng)然,如果只有一個(gè)條件,可以設(shè)計(jì)成2條SQL,比如:
復(fù)制代碼 代碼如下:

DECLARE @i INT;
SET @i=100
IF @i IS NOT NULL
    SELECT * FROM aaa WHERE age = @i
ELSE
    SELECT * FROM aaa

但是,如果條件多了,SQL數(shù)目也變得更多,所以建議用EXEC的方案

相關(guān)文章

最新評(píng)論