程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 數據庫知識 >> 其他數據庫知識 >> MSSQL >> SQL中WHERE變量IS NULL前提招致全表掃描成績的處理辦法

SQL中WHERE變量IS NULL前提招致全表掃描成績的處理辦法

編輯:MSSQL

SQL中WHERE變量IS NULL前提招致全表掃描成績的處理辦法。本站提示廣大學習愛好者:(SQL中WHERE變量IS NULL前提招致全表掃描成績的處理辦法)文章只能為提供參考,不一定能成為您想要的結果。以下是SQL中WHERE變量IS NULL前提招致全表掃描成績的處理辦法正文



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的查詢窗口裡,右擊窗口隨意率性地位,選擇“包括現實的履行籌劃”:

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

測試成果以下:

可以看到,即便@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的計劃

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved