优化SQL 语句 in 和not in 的替代方案

摘要:
用IN编写SQL的优点是易于编写和理解,这更适合现代软件开发风格。可以看出,至少还有一个使用IN的SQL转换过程。可以成功转换常规SQL,但无法转换具有分组统计信息的SQL。使用联接连接选项:SETSTATISTISTIMEONGO--备份数据插入bakInfoselecid、PName、remark、update、uptefromasstwhere。bakIDisnullandt upstate=0GOSETSTATISTISTICS关闭;此操作的执行时间:SQLServer分析和编译时间:CPU时间=247 ms,已用时间=247毫秒。可以看出,使用join方案的执行时间比使用notin和in的执行时间短得多

用IN写出来的SQL的优点是比较容易写及清晰易懂,这比较适合现代软件开发的风格。

但是用IN的SQL性能总是比较低的,从SQL执行的步骤来分析用IN的SQL与不用IN的SQL有以下区别:

SQL试图将其转换成多个表的连接,如果转换不成功则先执行IN里面的子查询,再查询外层的表记录,如果转换成功则直接采用多个表的连接方式查询。由此可见用IN的SQL至少多了一个转换的过程。一般的SQL都可以转换成功,但对于含有分组统计等方面的SQL就不能转换了。 推荐在业务密集的SQL当中尽量不采用IN操作符

NOT IN 此操作是强列推荐不使用的,因为它不能应用表的索引。推荐用NOT EXISTS 或(外连接+判断为空)方案代替

  在数据库中有两个表,一个是当前表Info(id,PName,remark,impdate,upstate),一个是备份数据表bakInfo(id,PName,remark,impdate,upstate),将当前表数据备份到备份表去,就涉及到not in 和in 操作了:

  首先,添加10万条测试数据

优化SQL 语句 in 和not in 的替代方案第1张优化SQL 语句 in 和not in 的替代方案第2张代码
 1 create procedure AddData
 2 as
 3 declare @id int 
 4 set @id=0
 5 while(@id<100000)
 6 begin
 7     insert into dbo.Info(id,PName,remark,impdate,upstate)
 8     values(@id,convert(varchar,@id)+'0','abc',getdate(),0)
 9     set @id=@id+1
10 end 
11 
12 exec AddData

  使用not in 和in操作:

SET STATISTICS TIME ON
GO
--备份数据
insert into bakInfo(id,PName,remark,impdate,upstate)
select id,PName,remark,impdate,upstate from dbo.Info
where id not in(select id from dbo.bakInfo)
GO
SET STATISTICS TIME OFF

  此操作执行时间:

SQL Server 分析和编译时间:
CPU 时间 = 0 毫秒,占用时间 = 3 毫秒。
SQL Server 执行时间:
CPU 时间 = 453 毫秒,占用时间 = 43045 毫秒。
(100000 行受影响)
SQL Server 分析和编译时间:
CPU 时间 = 0 毫秒,占用时间 = 1 毫秒。
--更改当前表状态
update  Info set upstate=1 where id in(select id from dbo.bakInfo)

  此操作执行时间:

SQL Server 分析和编译时间:
CPU 时间 = 62 毫秒,占用时间 = 79 毫秒。
SQL Server 执行时间:
CPU 时间 = 188 毫秒,占用时间 = 318 毫秒。
(100000 行受影响)
SQL Server 分析和编译时间:
CPU 时间 = 0 毫秒,占用时间 = 1 毫秒。
--删除当前表数据
delete from Info where upstate=1 and id in(select id from dbo.bakInfo)

  此操作执行时间:

SQL Server 分析和编译时间:
CPU 时间 = 183 毫秒,占用时间 = 183 毫秒。
SQL Server 执行时间:
CPU 时间 = 187 毫秒,占用时间 = 1506 毫秒。
(100000 行受影响)
SQL Server 分析和编译时间:
CPU 时间 = 0 毫秒,占用时间 = 1 毫秒。

  使用join连接替代方案:

SET STATISTICS TIME ON
GO
--备份数据
insert into bakInfo(id,PName,remark,impdate,upstate)
select id,PName,remark,impdate,upstate from
(SELECT     Info.id,Info.PName, Info.remark, Info.impdate,Info.upstate, bakInfo.id AS bakID
FROM         Info left JOIN
bakInfo ON Info.id = bakInfo.id ) as t
where t.bakID is null and t.upstate=0
GO
SET STATISTICS TIME OFF;


  此操作执行时间:

SQL Server 分析和编译时间:
CPU 时间 = 247 毫秒,占用时间 = 247 毫秒。
SQL Server 执行时间:
CPU 时间 = 406 毫秒,占用时间 = 475 毫秒。
(100000 行受影响)
SQL Server 分析和编译时间:
CPU 时间 = 0 毫秒,占用时间 = 1 毫秒。

 

--更改当前表状态
update Info set upstate=1
FROM         Info INNER JOIN
bakInfo ON Info.id = bakInfo.id

  此操作执行时间:

SQL Server 分析和编译时间:
CPU 时间 = 4 毫秒,占用时间 = 4 毫秒。
SQL Server 执行时间:
CPU 时间 = 219 毫秒,占用时间 = 259 毫秒。
(100000 行受影响)
SQL Server 分析和编译时间:
CPU 时间 = 0 毫秒,占用时间 = 1 毫秒。

--删除当前表数据
delete from Info
FROM         Info INNER JOIN
bakInfo ON Info.id = bakInfo.id
where  Info.upstate=1

  此操作执行时间:

SQL Server 分析和编译时间:
CPU 时间 = 177 毫秒,占用时间 = 177 毫秒。
SQL Server 执行时间:
CPU 时间 = 219 毫秒,占用时间 = 550 毫秒。
(100000 行受影响)
SQL Server 分析和编译时间:
CPU 时间 = 0 毫秒,占用时间 = 1 毫秒。

 

  可以看出使用join方案比使用not in 和in执行时间要短很多了优化SQL 语句 in 和not in 的替代方案第3张

 

免责声明:文章转载自《优化SQL 语句 in 和not in 的替代方案》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇解决来QQ消息后歌曲音量降低问题Qt Creator介绍下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

如何优化mysql查询速度

1、应尽量避免在 where 子句中使用!=或<>操作符,否则将引擎放弃使用索引而进行全表扫描。 2、对查询进行优化,应尽量避免全表扫描,首先应考虑在 where 及 order by 涉及的列上建立索引。 3、应尽量避免在 where 子句中对字段进行 null 值判断,否则将导致引擎放弃使用索引而进行全表扫描,如: select id fr...

【SQL Server学习笔记】联接提示、查询提示、表提示

SQL Server的查询优化器在select查询执行的时候产生一个高效的查询执行计划。如果优化器不能选择最优的计划,那么就需要检查查询计划、统计信息、支持的索引等,而通过使用提示可以改变优化器选择查询计划的工程,使优化器生成一个更好的执行计划。 1、联接提示 <join_hint> ::=      { LOOP | HASH | MERGE...

Windows Server 2008 R2(64位)下安装SQL Server 2005

目前笔者在做公司产品的UAT,在部署安装系统/软件时发现在安装SQL Server时容易出错,故总结如下 一、准备工作 在正式安装SQL Server 2005之前,我们要先启用ASP.NET和IIS,因为最终我们所安装的某些数据库功能将依赖于此,换句话说,如果我们不先安装这些东西,在我们正式安装数据库时,某些功能将无法安装,实际上,你根本无法选择安装对应...

数据库管理工具

NO1   Webyog | 管理和监控MySQL服务器的工具,官网:https://www.webyog.com/   SQLyog Ultimate V12.5 64bit 中文破解版(附注册码),下载地址:http://www.jb51.net/database/598306.html   SQLyong是一款功能强大的数据库管理程序,新版本增加更...

MS SQL的ISQL命令详解

MS SQL有简单方便又高效的管理界面,还用ISQL干什么?什么是ISQL工具?ISQL是用于交互式执行Transact-SQL语句和脚本的数据库工具,ISQL同微软的OSQL是差不多的工具。比如做一些高级的配置,还比如在客户端或远程执行数据库操作等工作时,ISQL是很方便的。比如入侵一台有SQL服务的主机,入侵SQL Server后都能干什么?最好的工具...

java基础知识--日期时间类

1.1 Date类  java.util.Date类 表示特定的瞬间,精确到毫秒。 继续查阅Date类的描述,发现Date拥有多个构造函数,只是部分已经过时,但是其中有未过时的构造函数可以把毫秒值转成日期对象。 public Date():分配Date对象并初始化此对象,以表示分配它的时间(精确到毫秒)。 public Date(long date)...