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

2023年 4月 18日 30.3k 0

复制代码 代码如下: 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开发时,这种写法是会

复制代码 代码如下: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)goCREATE INDEX idx_age ON aaa (age)GO

2、插入1万条测试数据:复制代码 代码如下:DECLARE @i INT;SET @i=0;WHILE @i<10000BEGIN  INSERT INTO aaa (name, age)VALUES(CAST(@i AS VARCHAR), @i)  SET @i=@i+1;ENDGO3、先开启执行计划显示:在SQL Server Management Studio的查询窗口里,右击窗口任意位置,选择“包含实际的执行计划”:

4、开始测试,用下面的SQL进行测试:复制代码 代码如下:DECLARE @i INT;SET @i=100SELECT * 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=100IF @i IS NOT NULL    SELECT * FROM aaa WHERE age = @iELSE     SELECT * FROM aaa

但是,如果条件多了,SQL数目也变得更多,所以建议用EXEC的方案

相关文章

Oracle如何使用授予和撤销权限的语法和示例
Awesome Project: 探索 MatrixOrigin 云原生分布式数据库
下载丨66页PDF,云和恩墨技术通讯(2024年7月刊)
社区版oceanbase安装
Oracle 导出CSV工具-sqluldr2
ETL数据集成丨快速将MySQL数据迁移至Doris数据库

发布评论