日期:2014-05-17  浏览次数:20551 次

关于 Parameter Sniff 想到的执行计划产生原理
本帖最后由 x_wy46 于 2014-01-24 00:03:16 编辑
问题来源于这个帖子,
http://bbs.csdn.net/topics/390698376
主要涉及的问题就是“存储过程执行慢,单独执行sql快”的疑问
参数嗅探的原因以及执行计划的生成过程之前也多多少少了解过一点
只是有些问题没有深入进去,今天遇到这个帖子的问题,
又测试了一下参数嗅探问题,同时又联想到sqlserver执行计划的产生规则
所以想总结点东西出来,大家一起讨论讨论

本文不是讨论参数嗅探的优化问题的,
只是借此来讨论参数嗅探及其“存储过程执行慢,单独执行sql快”产生的原因


先说明一下我的测试过程吧
create table TestSniff
(
id int,
cloumn1 varchar(50),
cloumn2 varchar(50),
cloumn3 varchar(50),
cloumn4 varchar(50)
)

truncate table TestSniff

--生成不均匀的数据分布,id=10的占整个表的10%
declare @i int 
set @i=1
while @i<10000
begin
if(@i%10=0)
begin
insert into TestSniff values(10,NEWID(),NEWID(),NEWID(),NEWID())
end
else
begin
insert into TestSniff values(@i,NEWID(),NEWID(),NEWID(),NEWID())
end;
set @i=@i+1
end

--id上建索引
create index index_id on testSniff(id)


set statistics profile on

set statistics io on

--因为10的分布占了整个表的10%,
--select * from TestSniff  where id=10;走表扫描是正确的选择
--为了验证,看了一下强制走索引,其代价也明显大于表扫描
--select * from TestSniff with(index(index_id)) where id=10;
select * from TestSniff  where id=10;


--这没问题,第一次执行,产生了一个缓存计划






create proc usp_testSniff(@i int)
as
declare @tmp int
set @tmp=@i
select * from TestSniff where id = @tmp --option(recompile)

--注意,执行select * from TestSniff  where id=10;之后没有清楚计划缓存
--先执行了select * from TestSniff  where id=10;
--那道理讲,缓存中已经有了执行计划,但是执行存储过程中,又重新编译了,
--选择了一个并不合理的执行计划
exec usp_testSniff 10






以前总是说,sqlserver生成计划的时候,是“完全一致”的sql才能重用计划
这里总算是又一次理解了这句话,
虽然缓存中有select * from TestSniff  where id=10;执行而存储的计划缓存
但是select * from TestSniff  where id=10;跟usp_testSniff中的sql是不完全一样的
所以执行存储过程前,还要重新编译




--再次执行
exec usp_testSniff 10

--这次才利用到计划缓存,可惜是上次生成的不合理的计划缓存









存储过程重编译也就算了,为什么没有生成一个合理的执行计划呢?
其实sqlserver生成执行计划的过程,当判断不了参数时,也是基于“猜测的”
比如说,他发现id上有索引,但是不知道具体的参数,那就根据猜测,走索引查找吧
尽管这种猜测可能不完全合理,
但是,对于大多数情况来说,比如执行其他9999个id不是10的查询,
走索引查找,是比较合理的
但是,正如这里,遇到数据分布不均匀的情况下,就悲剧了
那么这种猜测的根据是什么呢?


昨晚上也是用手机翻阅之前SQL_Beginner的一个精华帖
 http://bbs.csdn.net/topics/390667246

54楼发现了这个,这里借用一下

SELECT* from @tb OPTION(RECOMPILE)
--为什么 Estimated Rowcount为1024?
--因为总共就插入了1024行,插入之后再次评估它的行数,就能得它比较准确的值。

SELECT* from @tb where [C1] like '1' OPTION(RECOMPILE) 
--为什么 Estimated Rowcount为102.4?
--对于不模糊的like,Estimated Rowcount=total count *10%=1024*10%=102.4


Estimated Ro