前言

上一篇我们探讨了在静态语句中使用WHERE Column = @Param OR @Param IS NULL的问题,有对OPTION(COMPILE)的评论,那这节我们来探讨OPTION(COMPILE)的问题。

探讨OPTION(COMPILE)问题

在SQL SERVER中任何时候第一次调用存储过程时,此时存储过程将会被SQL SERVER优化且查询计划在内存中会被缓存。由于查询计划缓存,当运行相同的存储过程时,它都将使用相同的查询计划,从而无需每次运行时对同一存储过程进行优化和编译。因此,如果我们需要每天运行相同的存储过程若干次,那么可以节省大量的时间和硬件资源。

如果每次运行的存储过程中的在WHERE子句中具有相同的参数,则重复使用存储过程的相同查询计划是有意义的。但是,如果运行相同的存储过程,但是参数的值会改变呢?发生什么取决于参数的典型性。如果存储过程的参数的值从执行到执行相似,那么缓存的查询计划将正常工作,查询将按照执行最佳来。但是,如果参数不是典型的,那么被重用的缓存查询计划可能不是最优的,导致查询运行更慢,因为它使用的查询计划并不是真正为所使用的参数设计的。下面我们借助AdventureWorks2012示例数据库来用实例讲解上述所描述的情况。

DECLARE@AddressLine1 NVARCHAR(60) = NULL,@AddressLine2 NVARCHAR(60) = NULL,@City NVARCHAR(30) = NULL,@PostalCode NVARCHAR(15) = NULL,@StateProvinceID INT = NULL SET @City = 'Bothell'
SET @PostalCode = '98011'
SET @StateProvinceID = 79DECLARE @SQL NVARCHAR(MAX),@ColumnName VARCHAR(4000),@ParamDefinition NVARCHAR(500)SET @ColumnName = 'a.AddressID, a.AddressLine1, a.AddressLine2, a.City, a.StateProvinceID, a.PostalCode, a.rowguid'SET @SQL = 'SELECT ' + @ColumnName + ' FROM Person.Address AS a WHERE 1 = 1'IF (@AddressLine1 IS NOT NULL)SET @SQL = @SQL + ' AND a.AddressLine1 LIKE ''%'' + @AddressLine1 + ''%'''IF (@AddressLine2 IS NOT NULL)SET @SQL = @SQL + ' AND a.AddressLine2 LIKE ''%'' + @AddressLine2 + ''%'''IF (@City IS NOT NULL)SET @SQL = @SQL + ' AND a.City LIKE ''%'' + @City + ''%'''IF (@PostalCode IS NOT NULL)SET @SQL = @SQL + ' AND a.PostalCode LIKE ''%'' + @PostalCode + ''%'''IF (@StateProvinceID IS NOT NULL)SET @SQL = @SQL + ' AND a.StateProvinceID = @StateProvinceID' SET @ParamDefinition = N'@AddressLine1 NVARCHAR(60),
                         @AddressLine2 NVARCHAR(60),
                         @City NVARCHAR(30),
                         @PostalCode NVARCHAR(15),
                         @StateProvinceID INT'EXECUTE sp_executesql @SQL,@ParamDefinition,@AddressLine1  = @AddressLine1,@AddressLine2 = @AddressLine2,@City = @City,@PostalCode = @PostalCode,@StateProvinceID = @StateProvinceIDGO

我们运行上述查询1次,看到查询结果如下和计划缓存次数如下:

此时我们将外部变量StateProvinceID类型修改为SMALLINT,然后再来运行查询和缓存计划,此时会出现查询计划使用次数是为2,还是出现两条1呢?

此时我们再来将动态SQL中内部变量StateProvinceID类型修改为SMALLINT,此时会出现查询计划使用次数是为3,还是出现两条,次数分别为2和1呢?

由上可知,如果我们修改外部变量参数类型不会影响查询计划缓存即会达到重用目的,若修改动态SQL内部变量参数类型则不会重用查询计划缓存。

大多数情况下,我们可能不需要担心上述问题。但是,在某些情况下,假设从查询的执行到执行的参数变化很大,则会引起问题。如果我们确定存储过程通常运行正常,但有时运行缓慢,则很可能会看到上述问题。在这种情况下,我们可以做的是改变存储过程,并添加OPTION(RECOMPILE)选项。

添加此选项后,存储过程将始终重新编译自身,并在每次运行时创建一个新的查询计划。当然这会消除查询计划重用的好处,但确保了每次运行查询时都使用正确的查询计划。如果存储过程中有多个查询,那么它将重新编译存储过程中的所有查询,即使那些不受非典型参数影响的查询也是如此。

讲完OPTION(COMPILE),接下来我们讲讲如何创建性能稍高的存储过程。有些童鞋可能会创建如下存储过程。

CREATE PROC [dbo].[HighPerformanceExample]
(@AddressLine1 NVARCHAR(60) = NULL,@AddressLine2 NVARCHAR(60) = NULL,@City NVARCHAR(30) = NULL,@PostalCode NVARCHAR(15) = NULL,@StateProvinceID SMALLINT = NULL
)
AS
SET NOCOUNT ONSELECT a.AddressID, a.AddressLine1, a.AddressLine2, a.City, a.StateProvinceID, a.PostalCode, a.rowguid
FROM Person.Address AS a
WHERE (a.AddressLine1 = @AddressLine1 OR @AddressLine1 IS NULL) AND(a.AddressLine2 = @AddressLine2 OR @AddressLine2 IS NULL) AND(a.City = @City OR @City IS NULL) AND(a.PostalCode = @PostalCode OR @PostalCode IS NULL) AND(a.StateProvinceID = @StateProvinceID OR @StateProvinceID IS NULL)--或者
SELECT a.AddressID, a.AddressLine1, a.AddressLine2, a.City, a.StateProvinceID, a.PostalCode, a.rowguid
FROM Person.Address AS a
WHERE a.AddressLine1 = COALESCE(@AddressLine1, a.AddressLine1) ANDa.AddressLine2 = COALESCE(@AddressLine2, a.AddressLine2) ANDa.City = COALESCE(@City, a.City) ANDa.PostalCode = COALESCE(@PostalCode, a.PostalCode) ANDa.StateProvinceID = COALESCE(@StateProvinceID, a.StateProvinceID) --或者
SELECT a.AddressID, a.AddressLine1, a.AddressLine2, a.City, a.StateProvinceID, a.PostalCode, a.rowguid
FROM Person.Address AS a
WHERE a.AddressLine1 = CASE WHEN @AddressLine1 IS NULL THEN a.AddressLine1 ELSE @AddressLine1 ENDAND  a.AddressLine2 = CASE WHEN @AddressLine2 IS NULL THEN a.AddressLine1 ELSE @AddressLine2 ENDAND  a.City = CASE WHEN @City IS NULL THEN a.City ELSE @City ENDAND  a.PostalCode = CASE WHEN @PostalCode IS NULL THEN a.PostalCode ELSE @PostalCode ENDAND  a.StateProvinceID = CASE WHEN @StateProvinceID IS NULL THEN a.StateProvinceID ELSE @StateProvinceID END GOSET NOCOUNT OFF

上述无论怎样执行都将表现的非常糟糕。因为SQL SERVER不能将其很好地进行优化,如果这是由不同的参数组合产生,那么我们可能会得到一个绝对糟糕的计划。不难理解,当执行一个存储过程,并且还没有生成一个查询缓存计划。所以,管理员可能会更新统计信息或强制重新编译(或者,甚至重新启动SQL Server)来尝试解决此问题,但这些都不是最佳解决方案。OPTION(COMPILE)重新编译是个好东西,但是我们是不是像如下简单加上重新编译就可以了呢。

SELECT ...
FROM ...
WHERE ...
OPTION (RECOMPILE);

如果我们要使用重新编译,那么我们是否需要考虑以下两个问题呢?

如果我们知道一个特定的语句总是返回相同数量的行并使用相同的计划(并且我们已测试过并知道这一点),那么我们会正常创建存储过程并让计划得到缓存。

如果我们知道一个特定的语句从执行到执行是不一样的,最佳查询计划也会有所不同(我们也应该从执行多个测试样本中知道这一点),然后我们会如正常一样创建存储过程,然后使用OPTION(RECOMPILE)以确保语句的计划不会被存储过程缓存或保存。在每次执行时,存储过程将获得不同的参数,如此一来语句将在每次执行时得到一个新的计划。

为了实现这点,我们需要分析所查询的存储过程,例如在每个企业下有对应的用户,我们想象一下所呈现的UI界面,首先是所有用户,查询条件则是企业下拉框,然后是用户名或者员工工号等。当没有任何筛选条件时则走查询计划缓存,若选择企业,或者还选择了员工相关筛选条件则重新编译。类似如下存储过程。

CREATE PROC [dbo].[HighPerformanceExample]
(@AddressLine1 NVARCHAR(60) = NULL,@AddressLine2 NVARCHAR(60) = NULL,@City NVARCHAR(30) = NULL,@PostalCode NVARCHAR(15) = NULL,@StateProvinceID SMALLINT = NULL
)
AS
SET NOCOUNT ONDECLARE @SQL NVARCHAR(MAX),@ColumnName VARCHAR(4000),@ParamDefinition NVARCHAR(500),@Recompile  BIT = 1;SET @ColumnName = 'a.AddressID, a.AddressLine1, a.AddressLine2, a.City, a.StateProvinceID, a.PostalCode, a.rowguid'SET @SQL = 'SELECT ' + @ColumnName + ' FROM Person.Address AS a WHERE 1 = 1'IF (@StateProvinceID IS NOT NULL)SET @SQL = @SQL + ' AND a.StateProvinceID = @StateProvinceID' IF (@AddressLine1 IS NOT NULL)SET @SQL = @SQL + ' AND a.AddressLine1 LIKE @AddressLine1'IF (@AddressLine2 IS NOT NULL)SET @SQL = @SQL + ' AND a.AddressLine2 LIKE @AddressLine2'IF (@City IS NOT NULL)SET @SQL = @SQL + ' AND a.City LIKE @City'IF (@PostalCode IS NOT NULL)SET @SQL = @SQL + ' AND a.PostalCode LIKE @PostalCode'IF (@StateProvinceID IS NOT NULL)SET @Recompile = 0IF (PATINDEX('%[%_?]%',@AddressLine1) >= 4OR PATINDEX('%[%_?]%', @AddressLine2) = 0)AND (PATINDEX('%[%_?]%', @City) >= 4OR PATINDEX('%[%_?]%', @PostalCode) = 0)SET @Recompile = 0IF @Recompile = 1
BEGINSET @SQL = @SQL + N' OPTION(RECOMPILE)';
END;SET @ParamDefinition = N'@AddressLine1 NVARCHAR(60),
                         @AddressLine2 NVARCHAR(60),
                         @City NVARCHAR(30),
                         @PostalCode NVARCHAR(15),
                         @StateProvinceID SMALLINT'EXECUTE sp_executesql @SQL,@ParamDefinition,@AddressLine1  = @AddressLine1,@AddressLine2 = @AddressLine2,@City = @City,@PostalCode = @PostalCode,@StateProvinceID = @StateProvinceIDGOSET NOCOUNT OFF

总结

本节我们讲解了如何在存储过程中使用OPTION(COMPILE),并且使得存储过程性能达到最佳,我想这是根据实际场景分析存储过程相对来说首选和最佳的方法,以至于我们不必每次都重新编译。从而给我们长期更好的可扩展性。

本文转自Jeffcky博客园博客,原文链接:http://www.cnblogs.com/CreateMyself/p/8283105.html,如需转载请自行联系原作者

SQL Server-聚焦什么时候用OPTION(COMPILE)呢?相关推荐

  1. 在Windows Server 2016和SQL Server Always On可用性组上安装SQL Server 2019

    In this article, we will proceed with configuring a SQL Server Always On Availability Groups and per ...

  2. sql server 快照_添加新文章,删除文章,更改快照文件夹路径和SQL Server复制中的数据筛选器行

    sql server 快照 In the last articles, we have learned Configuring Snapshot and Transactional SQL Serve ...

  3. 如何对两个大型SQL Server数据库中的数据进行快速估计比较,以查看它们是否相等

    Bringing impactful analysis into a data always comes with challenges. In many cases, we rely on auto ...

  4. SQL Server 2016 SP1中的新功能和增强功能

    SQL Server 2016 SP1 is released as announced by Microsoft. It comes with a bunch of new features and ...

  5. aws rds监控慢sql_在AWS RDS SQL Server上的SSAS中部署表格数据库

    aws rds监控慢sql In this article, we are going to explore Analysis Service for AWS RDS SQL Server in de ...

  6. sql查询禁用缓存_如何在SQL Server 2017中启用和禁用身份缓存

    sql查询禁用缓存 Every data warehouse developer is likely to appreciate the significance of having surrogat ...

  7. 如何对SQL Server实例执行性能测试

    介绍 (Introduction) 目标听众 (Intended audience) This document is intended for application developers and ...

  8. sql数据库备份默认路径_在Linux上SQL Server中更改默认数据库文件和备份路径

    sql数据库备份默认路径 In a previous article, we explored the process to change default SQL dump file location ...

  9. SQL Server高级数据库管理员面试问答

    In this article, we will discuss a number of questions that you may be asked when applying to a seni ...

最新文章

  1. HTML5 网站大观:应用图片大背景的优秀 HTML5 网站作品
  2. 好奇了好久的「对象」,就这?
  3. if函数中的android,java - 我需要帮助将IF语句方法更改为Android Studio中的SWITCH情况 - 堆栈内存溢出...
  4. 国内代码托管平台(Git)
  5. 洛谷P1182 数列分段`Section II`
  6. DotnetCharting
  7. 重新想象 Windows 8 Store Apps (59) - 锁屏
  8. 使用git新增分支以及初始化分支等等一些列操作实战
  9. Java集合类的整理
  10. C#服务启动以及服务指令
  11. 读取jar中资源文件的问题
  12. DayDayUp:计算机技术与软件专业技术资格证书之《系统集成项目管理工程师》课程讲解之十大知识领域之4辅助—项目采购管理
  13. 遥控直升机主旋翼设定
  14. 绘制谢尔宾斯基三角形
  15. 2020浙江大学软件学院软件工程考研经验分享
  16. android美食app设计图,基于位置的Android美食地图应用的设计与实现
  17. Typora 建立内部跳转链接
  18. iOS7中计算UILabel中字符串的高度
  19. 【数值计算】数值解析--n元一次联立方程组:直接解法
  20. 国际青少年计算机技能大赛英语,Chinaspeaks国际青少年英语大赛_新浪博客

热门文章

  1. 关于ActionContext.getContext()的使用方法心得
  2. 东北黑木耳 微信公共帐号 销售 批发 分享 交流 东北鸿顺山特产品有限公司
  3. poladuo network 轻松解决Windows系统棘手问题
  4. OneGame V1.0.2 发布,让运营游戏不再是梦想
  5. shell清除日志小脚本
  6. 多层科目任意组合汇总报表的性能优化 (上)
  7. sql中varchar(n),nvarchar(n) 长度性能及所占空间分析
  8. C语言初学者简单语法综合练习
  9. Javascript控制Radio HTML控件
  10. BCH实用场景增加,Bitwage推出BCH工资单