1

我已经在管理工作室测试了这个查询,它执行得非常快(不到一秒)

declare @p_Message_0 varchar(3) = 'whh'
declare @p_CreatedTime_0 datetime = '2015-06-01'

SELECT count(1) FROM (SELECT * FROM [Logs](nolock) WHERE CONTAINS([Message], @p_Message_0) AND [CreatedTime]<@p_CreatedTime_0) t
SELECT t2.* FROM (SELECT t.*,ROW_NUMBER() OVER (ORDER BY Id DESC) as rownum FROM (SELECT * FROM [Logs](nolock) t WHERE CONTAINS([Message], @p_Message_0) AND [CreatedTime]<@p_CreatedTime_0) t) t2 WHERE rownum>0 AND rownum<=20

执行计划如下: 在此处输入图像描述

然后我将它移到 C# ado.net 中,它像这样运行

exec sp_executesql N'SELECT count(1) FROM (SELECT * FROM [Logs](nolock) WHERE CONTAINS([Message], @p_Message_0) AND [CreatedTime]<@p_CreatedTime_0) t
SELECT t2.* FROM (SELECT t.*,ROW_NUMBER() OVER (ORDER BY Id desc) as rownum FROM (SELECT * FROM [Logs](nolock) t WHERE CONTAINS([Message], @p_Message_0) AND [CreatedTime]<@p_CreatedTime_0) t) t2 WHERE rownum>0 AND rownum<=20',N'@p_Message_0 varchar(3),@p_CreatedTime_0 datetime',@p_Message_0='whh',@p_CreatedTime_0='2015-06-01'

这个运行速度非常慢(大约 30 秒)。执行计划如:在此处输入图像描述

我不知道是什么让这两个计划不同。Sql server 是带有 SP2 的 2008 R2,我已经尝试过参数提示和 OPTION (RECOMPILE),两者都不适合我。

4

1 回答 1

2

尝试更新统计信息。第一个使用带有今天日期的变量。变量不会被嗅探,所以你会得到一个猜测的分布。第二个使用参数。这可以被嗅探。

如果今天没有更新统计信息,SQL Server 将认为该日期不存在任何行,因此将在此基础上给出计划。例如一个嵌套循环计划,估计执行一次 TVF,但实际上执行了很多次。

又名升序日期问题

于 2015-06-01T05:25:32.117 回答