SQL Server中存储过程慢于SQL语句直接执行的原因

数据库 SQL Server
SQL Server数据库中,存储过程的执行总是要比SQL语句直接执行要慢,这究竟是为什么呢?本文将带您寻找答案。

SQL Server数据库中,存储过程的执行总是要比SQL语句直接执行要慢,这究竟是为什么呢?本文将带您寻找答案。

在SQL Server中有一个叫做 “Parameter sniffing”的特性。SQL Server在存储过程执行之前都会制定一个执行计划。在上面的例子中,SQL在编译的时候并不知道@thedate的值是多少,所以它在执行执行计划的时候就要进行大量的猜测。假设传递给@thedate的参数大部分都是非空字符串,而FACT表中有40%的thedate字段都是null,那么SQL Server就会选择全表扫描而不是索引扫描来对参数@thedate制定执行计划。全表扫描是在参数为空或为0的时候最好的执行计划。但是全表扫描严重影响了性能。

假设你第一次使用了Exec pro_ImAnalysis_daily @thedate=’20080312’那么SQL Server就会使用20080312这个值作为下次参数@thedate的执行计划的参考值,而不会进行全表扫描了,但是如果使用@thedate=null,则下次执行计划就要根据全表扫描进行了。

有两种方式能够避免出现“Parameter sniffing”问题:
<!--(1)通过使用declare声明的变量来代替参数:使用set @variable=@thedate的方式,将出现@thedate的sql语句全部用@variable来代替。
<!--(2) 将受影响的sql语句隐藏起来,比如:
<!-- a)      将受影响的sql语句放到某个子存储过程中,比如我们在@thedate设置成为今天后再调用一个字存储过程将@thedate作为参数传入就可以了。
<!-- b)      使用sp_executesql来执行受影响的sql。执行计划不会被执行,除非sp_executesql语句执行完。
<!-- c)      使用动态sql(”EXEC(@sql)”来执行受影响的sql。
采用(1)的方法改造例子中的存储过程,如下:

代码:

ALTER PROCEDURE [dbo].[pro_ImAnalysis_daily]@var_thedate VARCHAR(30) ASBEGIN    declare @THEDATE VARCHAR(30)    IF @var_thedate IS NULL    BEGIN       SET @var_thedate=CONVERT(VARCHAR(30),GETDATE()-1,112);    END      SET @THEDATE=@var_thedate;    DELETE FROM RPT_IM_USERINFO_DAILY WHERE THEDATE=@THEDATE;    INSERT RPT_IM_USERINFO_DAILY (THEDATE,ALLUSER,NEWUSER)    SELECT AA.THEDATE,ALLUSER,NEWUSER    FROM    ( ( SELECT THEDATE,COUNT(DISTINCT USERID) ALLUSER       FROM FACT       WHERE THEDATE=@THEDATE        GROUP BY THEDATE       ) AA       LEFT JOIN       (SELECT THEDATE,COUNT(DISTINCT USERID) NEWUSER        FROM FACT T1        WHERE NOT EXISTS(                         SELECT 1                         FROM FACT T2                         WHERE T2.THEDATE<@THEDATE                             AND T1.USERID=T2.USERID)              AND T1.THEDATE=@THEDATE        GROUP BY THEDATE        ) BB       ON AA.THEDATE=BB.THEDATE);GO
 

【编辑推荐】

SQL SERVER数据库中存储过程使用循环语句

Sql Server两个版本中显示所有表信息的语句

SQL Server中一个语句块实现多条语句插入的方法

SQL Server数据库用户创建的方法(使用SQL语句)

使用SQL语句查看SQL Server事务日志的方法

 

责任编辑:段燃 来源: 互联网
相关推荐

2010-11-12 09:58:34

SQL存储过程

2010-11-04 09:43:46

LINQ to SQL

2010-09-03 15:08:03

SQLselect语句

2010-09-07 11:41:24

SQL语句

2010-09-06 11:05:05

SQL SERVER语句

2010-09-25 16:21:41

SQL语句

2010-07-15 12:38:14

SQL Server存

2011-04-02 16:39:53

SQL Server查询

2011-03-24 13:38:47

SQL Server 存储分页

2010-09-06 11:46:03

SQL Server语句

2009-08-06 16:44:06

2010-04-29 14:06:40

Oracle SQL

2010-09-07 15:12:25

SQL语句优化

2010-11-12 09:18:13

SQL Server存

2009-11-05 18:07:33

Oracle导出sql

2011-07-14 13:38:34

2010-11-12 13:08:36

动态sql语句

2011-05-20 15:59:06

Oracle存储Sql语句

2010-11-10 13:03:15

SQL Server存

2010-09-06 13:41:42

sql server语句
点赞
收藏

51CTO技术栈公众号