源码网商城,靠谱的源码在线交易网站 我的订单 购物车 帮助

源码网商城

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

  • 时间:2020-05-04 16:37 编辑: 来源: 阅读:
  • 扫一扫,手机访问
摘要:SQL中WHERE变量IS NULL条件导致全表扫描问题的解决方法
[u]复制代码[/u] 代码如下:
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、建立测试用的表结构和索引:
[u]复制代码[/u] 代码如下:
CREATE TABLE aaa(id int IDENTITY, NAME VARCHAR(12), age INT) go CREATE INDEX idx_age ON aaa (age) GO
2、插入1万条测试数据:
[u]复制代码[/u] 代码如下:
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的查询窗口里,右击窗口任意位置,选择“包含实际的执行计划”: [img]http://files.jb51.net/file_images/article/201309/2013926153312656.jpg[/img] 4、开始测试,用下面的SQL进行测试:
[u]复制代码[/u] 代码如下:
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
测试结果如下: [img]http://files.jb51.net/file_images/article/201309/2013926153422925.jpg[/img] 可以看到,即使@i有值,不管@i IS NULL是放在前面还是放在后面,都无法用到age的索引,另外age=ISNULL(@i,age)也用不上索引 最终结论,SQL Server跟ORACLE一样,如果条件里加了 变量 IS NULL,都会导致全表扫描。 建议SQL改成:
[u]复制代码[/u] 代码如下:
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,比如:
[u]复制代码[/u] 代码如下:
DECLARE @i INT; SET @i=100 IF @i IS NOT NULL     SELECT * FROM aaa WHERE age = @i ELSE     SELECT * FROM aaa
但是,如果条件多了,SQL数目也变得更多,所以建议用EXEC的方案
  • 全部评论(0)
联系客服
客服电话:
400-000-3129
微信版

扫一扫进微信版
返回顶部