栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 系统运维 > 数据库 > MySQL > MsSql

SQL中WHERe变量IS NULL条件导致全表扫描问题的解决方法

MsSql 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

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的方案

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/169717.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号