国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 數據庫 > SQL Server > 正文

SQL中WHERE變量IS NULL條件導致全表掃描問題的解決方法

2024-08-31 01:00:27
字體:
來源:轉載
供稿:網友

復制代碼 代碼如下:


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開發時,這種寫法是會導致全表掃描的,用不上索引,不知道Sql Server里是否也是一樣呢,于是做一個簡單的測試
1、建立測試用的表結構和索引:

復制代碼 代碼如下:


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

2、插入1萬條測試數據:

復制代碼 代碼如下:


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、先開啟執行計劃顯示:
在SQL Server Management Studio的查詢窗口里,右擊窗口任意位置,選擇“包含實際的執行計劃”:

SQL中WHERE變量IS NULL條件導致全表掃描問題的解決方法

4、開始測試,用下面的SQL進行測試:

復制代碼 代碼如下:


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


測試結果如下:

SQL中WHERE變量IS NULL條件導致全表掃描問題的解決方法

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

最終結論,SQL Server跟ORACLE一樣,如果條件里加了 變量 IS NULL,都會導致全表掃描。

建議SQL改成:

復制代碼 代碼如下:


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


當然,如果只有一個條件,可以設計成2條SQL,比如:

復制代碼 代碼如下:


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

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

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 彰武县| 益阳市| 招远市| 林甸县| 新昌县| 乌鲁木齐县| 浦东新区| 泰宁县| 莱阳市| 竹溪县| 红原县| 黄浦区| 曲靖市| 尖扎县| 灵丘县| 广西| 临洮县| 简阳市| 南陵县| 苍南县| 元谋县| 张家川| 中宁县| 前郭尔| 双江| 宝兴县| 武平县| 麦盖提县| 当雄县| 方城县| 桓仁| 隆安县| 肥城市| 潞西市| 柳州市| 雷州市| 澄城县| 砀山县| 卓资县| 称多县| 普格县|