SQLSERVER为了确保返回正确的值,或者处于性能上的顾虑,有意不重用缓存在内存里的执行计划,而重新编译执行计划的这种行为,被称为重编译(recompile)。那么引发存储过程重编译的条件有哪一些呢?下面罗列了一些导致重编译(recompile)的条件:

– 对查询所引用的表或视图进行更改(ALTER TABLE 和 ALTER VIEW)。

– 对执行计划所使用的任何索引进行更改。

– 对执行计划所使用的统计信息进行更新,这些更新可能是从语句(如 UPDATE STATISTICS)中显式生成,也可能是自动生成的。

– 删除执行计划所使用的索引。

– 显式调用 sp_recompile。

– 对键的大量更改(其他用户对由查询引用的表使用 INSERT 或 DELETE 语句所产生的修改)。

– 对于带触发器的表,插入的或删除的表内的行数显著增长。

– 使用 WITH RECOMPILE 选项执行存储过程。

– 有些DBCC FREEPROCCACHE;分离、附加数据库、数据升级也会清除内存里缓存的执行计划

好了,切入到今天我们要关注的问题:临时表的数据变化导致存储过程重编译问题,其实临时表的数据变化导致存储过程重编译实质上是因为临时表的数据变化,导致了临时表统计信息的自动更新,从而引起的重编译。那么触发临时表的统计信息的更新的条件或阀值是什么呢?说来也简单,就是下面一个这个公式(n表示变更前临时表的数据记录数,确切的说是上一次采集统计信息时临时表的记录数)

Temporary table

1、 If n < 6, RT = 6.

2、 If 6 <= n <= 500, RT = 500.

3、 If n > 500, RT = 500 + 0.20 * n.

有个网友说存储过程中的临时表数据变更的阀值有问题:他的原话如下

If n < 6, Recompilation threshold = 6.

If 6 <= n <= 500, Recompilation threshold = 500.

上面这两个区间没有问题。但是大于500的之后,根本就不是变化大于20%之后再重编译。看了他提出的问题,其实我也不是特肯定,毕竟没有实际验证过。实践才是检验整理的唯一标准,那么我们就开始做实验吧,首先准备一下测试环境(Microsoft SQL Server 2008 (RTM) – 10.0.1600.22 (X64) ).脚本如下所示:

USE MyDBA;

GO

IF EXISTS(SELECT 1 FROM sys.sysobjects WHERE id=object_id(N'[dbo].[TEST]') AND OBJECTPROPERTY(id, N'IsTable')=1 )

BEGIN

DROP TABLE dbo.TEST;

CREATE TABLE TEST

(

ID INT IDENTITY(1, 1) ,

NAME VARCHAR(40)     ,

CONSTRAINT PK_TEST PRIMARY KEY(ID)

)

END

GO

INSERT INTO TEST VALUES(NEWID())

GO 10000

CREATE PROCEDURE Usp_Recompile_TEST(@Index INT)

AS

BEGIN

CREATE TABLE #T(ID   INT , NAME VARCHAR(40));

INSERT INTO #T SELECT ID, NAME FROM TEST WHERE ID <=@Index;

SELECT  m.* FROM #T m INNER JOIN TEST n ON m.ID = n.ID

END

GO

准备好测试环境后,那么此时我们打开SQL Server工具SQL Server Profiler,选择“SP:Recompile”和“SP:Complete”事件,然后取消一些选择列,仅仅选择一些需要的列,例如 EventClass、TextData等。如下所示

开启Profile跟踪后,我们打开一个会话窗口,勾选“包括实际的执行计划”,然后再窗口执行下面SQL语句

EXEC dbo.Usp_Recompile_TEST 1;

如下所示,实际的执行计划中,我们看到“估计行数”和“实际行数”是一致的。

EXEC dbo.Usp_Recompile_TEST 2;

EXEC dbo.Usp_Recompile_TEST 6;

执行上面两个语句,我们会发现“估计行数”与“实际行数”开始出现偏差,因为数据库对临时表#T没有最新的统计信息,还是上一次收集的统计信息时的数据(1行数据)

EXEC dbo.Usp_Recompile_TEST 7; 此时已经触发了对临时表统计信息的采集更新(请见后面阐述)。

EXEC dbo.Usp_Recompile_TEST 130;

EXEC dbo.Usp_Recompile_TEST 500;

EXEC dbo.Usp_Recompile_TEST 506;

EXEC dbo.Usp_Recompile_TEST 507;

那么执行上面SQL语句,130我们确信不会导致临时表#T去更新统计信息,501会触发#T表的统计信息更新吗? 如果不会触发,那么确切的值是多少呢?答案是507,如下截图所示:

想必有些人会说,我实验的结果不一样哦(啪啦啪啦说一大堆),那么你是否真正的理解了下面公式呢? n表示临时表变跟前的记录数(确切的说是统计信息采集时的记录数),后面的RT表示变跟的记录数。

Temporary table

1、 If n < 6, RT = 6.

2、 If 6 <= n <= 500, RT = 500.

3、 If n > 500, RT = 500 + 0.20 * n.

由于我第一次执行的是EXEC dbo.Usp_Recompile_TEST 1,那么数据库的记录数为1,那么1+ 6 =7; 也就是上图EXEC dbo.Usp_Recompile_TEST 7时才触发临时表#T的统计信息更新,而为什么是507(7+500=507)呢,因为最后一次统计信息的采集,临时表#T的记录数为7 ,所以7+500=507,是否有点不解,那么你按我这个SQL执行一遍,然后用Profile跟踪、你会看到下面结果,如果还不太明白,结合截图好好理解一下:

DBCC FREEPROCCACHE;

EXEC dbo.Usp_Recompile_TEST 2;

EXEC dbo.Usp_Recompile_TEST 6;

EXEC dbo.Usp_Recompile_TEST 7;

EXEC dbo.Usp_Recompile_TEST 8;

如果还没有理解的话,我的表达能力已到极限了,自己再好好琢磨一下吧! 那么接下来才是我们重点想要验证、测试的。

DBCC FREEPROCCACHE;

EXEC dbo.Usp_Recompile_TEST 501;

此时临时表#T的记录数为501,那么当临时表#T里面的记录数变更了多少时,才会触发统计信息的更新呢? 由于是插入,那么根据公式应该是501 + (500 + 0.2*501) = 1101.2 ,那么应该是1101,即使是1100也不会变化。下面SQL Server Profile可以验证我们的推测

EXEC dbo.Usp_Recompile_TEST 1100;

EXEC dbo.Usp_Recompile_TEST 1101;

如果我们继续使用该存储过程,那么当参数为什么值时才会触发统计信息更新呢? 1101 +(500+0.2*1101)=1821.2,也就是说必须是1821才会触发统计信息更新,下面SQL Server Profile的截图也验证了我们的推测。

EXEC dbo.Usp_Recompile_TEST 1300;

EXEC dbo.Usp_Recompile_TEST 1320;

EXEC dbo.Usp_Recompile_TEST 1321;

EXEC dbo.Usp_Recompile_TEST 1820;

EXEC dbo.Usp_Recompile_TEST 1821;

所以综上述实验验证,SQL SERVER 临时表导致存储过程重编译(recompile)的那些阀值确实是正确的,也是没有问题的。当然如有疏漏或不对的地方,敬请指出。

java编译sql存过_SQL SERVER 临时表导致存储过程重编译(recompile)的一些探讨相关推荐

  1. mysql 存储过程 compile_SQLSERVER临时表导致存储过程重编译(recompile)的一些探讨_MySQL...

    SQLSERVER为了确保返回正确的值,或者处于性能上的顾虑,有意不重用缓存在内存里的执行计划,而重新编译执行计划的这种行为,被称为重编译(recompile).那么引发存储过程重编译的条件有哪一些呢 ...

  2. 导致存储过程重新编译的原因

    存储过程包含一组复杂的SQL语句,使生成存储过程的执行计划的代价有些高.因此通常重用存储过程的执行计划来代替生成新计划是有利的.但是有时候现有的计划可能不适用或者在重用期间可能不能提供最佳的处理策略. ...

  3. mysql表变量临时表_sql server 临时表详细讲解及简单示例

    一.概述 在sql server里临时表存储在TempDB库中,TempDB是一个系统数据库,它只有Simple恢复模式,也是最小日志记录操作.主要用于存放局部临时表,全局临时表,表变量,都是基于临时 ...

  4. java sql 创建触发器_SQL Server创建触发器

    在本教程中,将学习如何使用SQL Server CREATE TRIGGER语句来创建新的触发器. SQL Server CREATE TRIGGER语句简介 CREATE TRIGGER语句用于创建 ...

  5. sql tempdb清理_SQL Server 2019内存优化的TempDB元数据

    sql tempdb清理 In this article, I will walk you through the new feature in SQL Server 2019, memory-opt ...

  6. sql tempdb清理_SQL Server 2019中的内存优化的TempDB元数据

    sql tempdb清理 介绍 (Introduction) In-memory technologies are one of the greatest ways to improve perfor ...

  7. php执行sql内存溢出_SQL Server 2017:SQL排序,溢出,内存和自适应内存授予反馈

    php执行sql内存溢出 This article explores SQL Sort, Spill, Memory and Adaptive Memory Grant Feedback mechan ...

  8. server sql 本月最后一天_SQL Server 获取最后一天(指定时间的月最后一天日期)...

    /* author OceanHo @ 2015-10-23 10:14:21 获取指定时间字符串指定日期的月最后一天日期 */ IF OBJECT_ID('get_LastDayDate') IS ...

  9. java解析sql查询字段_sql解析json格式字段 如何获取json中某个字段的值?

    java将json数据解析为sql语句?小编给你倒一杯热水.可你惦记着其他饮料,所以你将它放置一旁.等你想起那杯水时,可惜它已经变得冰冷刺骨. 图片中是json数据,每个数据的开头都有表名称,操作类型 ...

最新文章

  1. nodejs开发部署工具,守护进程
  2. 035_Breadcrumb面包屑
  3. python当中pip使用_python
  4. access 战地1不加入ea_战地1正式加入origin access
  5. mysql_num_rows+报错_错误:警告:mysql_num_rows()期望参数1为资源,在第19行的C:\ xampp...
  6. OpenCV像素二位数组数据(矩阵)生成图片
  7. 智能优化算法应用:基于GWO优化的指数熵图像多阈值分割 - 附代码
  8. 抢票软件开发(二) 模拟登录
  9. celeste第二章_《蔚蓝(Celeste)》全水晶之心收集攻略
  10. 目标检测正负样本区分和平衡策略总结
  11. “过关斩将”的第一台电脑
  12. python画长方形的代码_python使用turtle画一个三角形、正方形(矩形或四边形)
  13. 数据结构-malloc申请动态空间-链表的创建
  14. SSH机试顺丰搬家预约信息查询参考
  15. getsockopt/setsockopt函数用法【转】
  16. java 编程 网易公开课,java毕业设计_springboot框架的社区居民健康档案管理实现
  17. 学习笔记 | 数据结构和算法 知识点思维导图
  18. python中time库引用不正确的_time库的引用
  19. 坯子库曲面推拉教程_为了学妹,掉几根头发算什么!学长熬夜整理的SU插件合集大放送!...
  20. MYSQL中查询怎么判断一个字段包含英文?

热门文章

  1. 常见SQL Server 2000漏洞及其相关利用2
  2. nyoj138 哈希的简单应用(查找)
  3. java8(2)--- Stream API
  4. c# 封装“度分秒”与弧度之间的转换 以及datagridview控件的应用
  5. 软件测试技术第一次试验之——JUnit的安装与使用
  6. Elasticsearch-2.3.x填坑之路
  7. BestCoder4 1002 Miaomiao's Geometry (hdu 4932) 解题报告
  8. Android 2.3发短信详细流程
  9. 设计人的33个好习惯
  10. Qomo OpenProject Field Test 1发布!