如何track存储过程的编译次数
有个script我们很熟悉,是用来去查找当前SQL
Server中哪些存储过程变重编译的次数最多的:
--Gives you the top 25 stored procedures that have been recompiled.
select top 25 sql_text.text, sql_handle, plan_generation_num, execution_count,
dbid, objectid
into DMV_Top25_Recompile_Commands
from sys.dm_exec_query_stats a
cross apply sys.dm_exec_sql_text(sql_handle) as sql_text
where plan_generation_num >1
order by plan_generation_num desc
go
那么,这个脚本究竟是记录什么情况下的存储过程recomile呢?
我们在SQL Server上创建一个这样的store
procedure:
create proc aaa
as
select plan_generation_num,* FROM DMV_Top25_Recompile_Commands where plan_generation_num > 2
然后准备好用这个脚本来返回plan_generation_num的值
select top 25 sql_text.text, sql_handle, plan_generation_num, execution_count,
dbid, objectid
from sys.dm_exec_query_stats a
cross apply sys.dm_exec_sql_text(sql_handle) as sql_text
where sql_text.text like '%aaa%'
order by plan_generation_num desc
Exec aaa之后的脚本返回结果:
这里的第六行结果集就是我们的存储过程aaa。这时的plan_generation_num值显示为1.
接下来我们mark recompile:
sp_recompile aaa
然后再次执行 exec aaa
使用脚本查询:
这里看到存储过程重编译以后,plan_generation_num的值并没有增加。
那为什么我们还会使用这样的脚本来返回重编译次数很多的存储过程呢?
接下来我们再次将存储过程mark recompile,然后直接使用脚本查询:
这时,我们发现该存储过程的plan 和text已经从DMV中移除了。看起来sp_recompile会直接将cache中缓存的执行计划和语句直接标识成不可用。因此DMV中就没有相关的记录了。
这就是说,存储过程标识重编译这种模式导致的重编译,从DMV里面是没有办法跟踪的。
那么从性能监视器的计数器 “sp
recompilation/sec”里面能不能跟踪到呢?
我们反复执行:
sp_recompile aaa
exec aaa
性能监视器中一直显示为0
那么plan_generation_num的值究竟是什么含义呢?BOL中的解释很简单:
A sequence number that can be used to distinguish between instances of plans after a recompile.
中文版的含义为:可用于在重新编译后区分不同计划实例的序列号。
这里并没有说明如何去计算的序列号。我们从另一篇英文的blog中找到了更加详细的说明:
There are a lot of interesting columns in P and S, especially in S, and here I will only discuss what I have learned about plan_generation_num in S. SQL Server 2005 treats the compiled plan for a stored
procedure as an array of subplans, one for each query statement. If an individual subplan needs recompilation, it does so without causing the whole plan to recompile. In doing so, SQL Server increments the plan_generation_num on the subplan record to be 1
+ MAX(plan_generation_num for all subplans). The general distribution of plan_generation_num among all subplans for a given plan is such that it has multiple of 1's and distinct numbers > 1. That is because all subplans start with 1 as their plan_generation_num.
Appendix A is the query for learning plan_generation_num.
http://lfsean.blogspot.com/2008/02/understanding-sql-plangenerationnum.html
这部分说明简单的来说,就是只要存储过程中有一条语句发生重编译,这个plan_generation_num值就会+1.这里并没有说是整个存储过程重编译的时候,这个值会+1.
接下来我们修改测试存储过程aaa:
Alter TABLE aaa_table(
[text] [nvarchar](max) NULL,
[sql_handle] [varbinary](64) NOT NULL,
[plan_generation_num] [bigint] NOT NULL,
[execution_count] [bigint] NOT NULL,
[dbid] [smallint] NULL,
[objectid] [int] NULL
) ON [PRIMARY]
insert into aaa_table select * from DMV_Top25_Recompile_Commands where 1=2
insert into aaa_table select * from DMV_Top25_Recompile_Commands where 1=2
insert into aaa_table select * from DMV_Top25_Recompile_Commands where 1=2
insert into aaa_table select * from DMV_Top25_Recompile_Commands where 1=2
insert into aaa_table select * from DMV_Top25_Recompile_Commands where 1=2
然后我们执行存储过程,收集profiler trace,同时继续监控性能监视器
开始重新执行存储过程aaa
这里我们可以看到sp recompilation/sec立刻变成了7。
Profiler trace中可以看到每条insert语句上都触发了一个sp:recompile
脚本的查询结果:
可以看到plan_generation_num的值增加到6了。
为什么这样写存储过程会导致重编译?http://support.microsoft.com/kb/243586 这篇文章中列举了多种会导致存储过程重编译的情况:
aaa这个存储过程符合这个条件:
The procedure interleaves Data Definition Language (DDL) and Data Manipulation Language (DML) operations.
因此我们的结论是,使用这个脚本去查询重编译次数高的存储过程是没有错的,但是这个脚本并不包含由于sp_recompile已经定义存储过程时使用了with
recompile的选项而导致的存储过程重编译的情况。
如何track存储过程的编译次数的更多相关文章
- SQL SERVER 临时表导致存储过程重编译(recompile)的一些探讨
SQLSERVER为了确保返回正确的值,或者处于性能上的顾虑,有意不重用缓存在内存里的执行计划,而重新编译执行计划的这种行为,被称为重编译(recompile).那么引发存储过程重编译的条件有哪一些呢 ...
- sqlserver 存储过程中使用临时表到底会不会导致重编译
曾经在网络上看到过一种说法,SqlServer的存储过程中使用临时表,会导致重编译,以至于执行计划无法重用, 运行时候会导致重编译的这么一个说法,自己私底下去做测试的时候,根据profile的跟踪结果 ...
- SQL Server 查看存储过程执行次数的方法
今天老大提出一个需求,想查看数据库存储过程执行的次数,以前没有接触过,于是网上找了下,发现还真有! 不废话,贴出来sql语句,直接执行即可看到结果: use master select text,ex ...
- 关于T-SQL重编译那点事,WITH RECOMPILE和OPTION(RECOMPILE)区别仅仅是存储过程级重编译和SQL语句级重编译吗
本文出处:http://www.cnblogs.com/wy123/p/6262800.html 在考虑重编译T-SQL(或者存储过程)的时候,有两种方式可以实现强制重编译(前提是忽略导致重编译的 ...
- 理解性能的奥秘——应用程序中慢,SSMS中快(2)——SQL Server如何编译存储过程
本文属于<理解性能的奥秘--应用程序中慢,SSMS中快>系列 接上文:理解性能的奥秘--应用程序中慢,SSMS中快(1)--简介 本文介绍SQL Server如何编译存储过程并使用计划缓存 ...
- WITH RECOMPILE和OPTION(RECOMPILE)区别仅仅是存储过程级重编译和SQL语句级重编译吗
在考虑重编译T-SQL(或者存储过程)的时候,有两种方式可以实现强制重编译(前提是忽略导致重编译的其他因素的情况下,比如重建索引,更新统计信息等等), 一是基于WITH RECOMPILE的存储过程级 ...
- Oracle-检查原因并重新编译无效的存储过程
1.查看存储过程编译无效的原因 show errors procedure hr.getperson; 2.指定一个存储过程进行编译 alter procedure hr.getperson com ...
- oracle 存储过程 包 【转】
一.为什么要用存储过程? 如果在应用程序中经常需要执行特定的操作,可以基于这些操作简历一个特定的过程.通过使用过程可以简化客户端程序的开发和维护,而且还能提高客户端程序的运行性能. 二.过程的优点? ...
- 小白学习mysql之存储过程的优劣分析以及接入控制
存储过程的优劣 存储过程是一组实现特定功能的SQL语句集合,存储过程一经编译便存储在了服务器上,可以通过调用存储过程的名字以及传入相应的参数来使用存储过程.要高层次的掌握存储过程,不能觉得依葫芦画瓢, ...
随机推荐
- 【Cloud Foundry】Could Foundry学习(三)——Router
在阅读的过程中有不论什么问题.欢迎一起交流 邮箱:1494713801@qq.com QQ:1494713801 一.概述 Router组件在Cloud Foundry中是对全部进来的Reque ...
- poj 3311 状压DP
经典TSP变形 学到:1.floyd O(n^3)处理随意两点的最短路 2.集合的位表示,我会在最后的总结出写出.注意写代码之前一定设计好位的状态.本题中,第0位到第n位分别代表第i个城市,1是已经 ...
- swift 笔记 (十八) —— 扩展
扩展 扩展能够让我们给一个已有的类.结构体.枚举等类型加入�新功能,包含属性和方法,甚至是构造器,下标,支持协议等等... 甚至是我们拿不到源码的类.结构体.枚举,我们依旧能够给它加扩展... 看到这 ...
- json与jsonp区别浅析(json才是目的,jsonp只是手段) (转)
一言以蔽之,json返回的是一串数据:而jsonp返回的是脚本代码(包含一个函数调用): JSON其实就是JavaScript中的一个对象,跟var obj={}在质上完全一样,只是在量上可以无限扩展 ...
- 行为驱动开发(BDD)
行为驱动开发(BDD) 引言 BDD是对TDD理念的扩展.BDD强调有利害关系的技术团体和非技术团队都要参与到软件开发过程中.可以把它看成一种强调团体间合作的敏捷方法.大多数采用某种敏捷方法的团队最终 ...
- import android.provider.Telephony cannot be resolved
android.provider.Telephony is hidden. http://androidxref.com/4.0.3_r1/xref/frameworks/base/core/java ...
- ZOJ Problem Set - 3829Known Notation(贪心)
ZOJ Problem Set - 3829Known Notation(贪心) 题目链接 题目大意:给你一个后缀表达式(仅仅有数字和符号),可是这个后缀表达式的空格不幸丢失,如今给你一个这种后缀表达 ...
- Linux 下 Error: Could not find or load main class Hello
在linux下写了一个很easy的Hello world程序,编译执行居然报错:Error: Could not find or load main class Hello 最后发现是CLASSPAT ...
- 新项目架构从零开始(三)------基于简单ESB的服务架构
这几个月一直在修改架构,所以迟迟没有更新博客. 新的架构是一个基于简单esb的服务架构,主要构成是esb服务注册,wcf服务,MVC项目构成. 首先,我门来看一看解决方案, 1.Common 在Com ...
- Socket的错误码和描述(中英文翻译)
Socket的错误码和描述(中英文翻译) //下面是Socket Error的错误码和描述: Socket error 0 - Directly send error Socket error 10 ...