有时候,我们需要查看存储过程的执行计划,那么我们有什么方式获取存储过程的历史执行计划或当前的执行计划呢? 下面总结一下获取存储过程的执行计划的方法。

1:我们可以通过下面脚本查看存储过程的执行计划,但是有时候,你会发现这种方式并不总是能够获取到存储过程的执行计划。

SELECT

        d.object_id ,

        DB_NAME(d.database_id) DBName ,

        OBJECT_NAME(object_id, database_id) 'SPName' ,

        d.cached_time ,

        d.last_execution_time ,

        d.total_elapsed_time/1000000    AS total_elapsed_time,

        d.total_elapsed_time / d.execution_count/1000000 

                                        AS [avg_elapsed_time] ,

        d.last_elapsed_time/1000000     AS last_elapsed_time,

        d.execution_count ,

        d.total_physical_reads ,

        d.last_physical_reads ,

        d.total_logical_writes ,

        d.last_logical_reads ,

        et.text SQLText ,

        eqp.query_plan executionplan

FROM    sys.dm_exec_procedure_stats AS d

CROSS APPLY sys.dm_exec_sql_text(d.sql_handle) et

CROSS APPLY sys.dm_exec_query_plan(d.plan_handle) eqp

WHERE   OBJECT_NAME(object_id, database_id) = 'xxxx'

ORDER BY [total_worker_time] DESC;

有时候使用这种方式并不能获取存储过程的执行计划,脚本查询出来的结果,query_plan字段为NULL值,那么为什么是NULL值呢?这个是因为有一些限制或条件的缘故,官方文档的解释如下:

Under the following conditions, no Showplan output is returned in the query_plan column of the returned table for sys.dm_exec_query_plan:

·         If the query plan that is specified by using plan_handle has been evicted from the plan cache, the query_plan column of the returned table is null. For example, this condition may occur if there is a time delay between when the plan handle was captured and when it was used with sys.dm_exec_query_plan.

·         Some Transact-SQL statements are not cached, such as bulk operation statements or statements containing string literals larger than 8 KB in size. XML Showplans for such statements cannot be retrieved by using sys.dm_exec_query_plan unless the batch is currently executing because they do not exist in the cache.

·         If a Transact-SQL batch or stored procedure contains a call to a user-defined function or a call to dynamic SQL, for example using EXEC (string), the compiled XML Showplan for the user-defined function is not included in the table returned by sys.dm_exec_query_plan for the batch or stored procedure. Instead, you must make a separate call to sys.dm_exec_query_plan for the plan handle that corresponds to the user-defined function.

When an ad hoc query uses simple or forced parameterization, the query_plan column will contain only the statement text and not the actual query plan. To return the query plan, call sys.dm_exec_query_plan for the plan handle of the prepared parameterized query. You can determine whether the query was parameterized by referencing the sql column of the sys.syscacheobjects view or the text column of the sys.dm_exec_sql_text dynamic management view.

在以下情况下,sys.dm_exec_query_plan的返回表的query_plan列为空值(query_plan列中未返回Showplan输出):

·         通过使用plan_handle查询指定的查询计划(query plan),如果plan_handle已从计划缓存中踢出(逐出),返回的表的query_plan列为null。 例如,如果在捕获计划句柄与将其与sys.dm_exec_query_plan一起使用之间存在时间延迟,则可能会出现这种情况。

·         有些Transact-SQL语句不会cached,例如大容量操作语句(bulk operation statements)或包含大于8 KB的字符串大小的SQL语句。无法使用sys.dm_exec_query_plan检索此类语句的XML Showplans,除非批处理当前正在执行,因为它们不存在于缓存中。

·         如果Transact-SQL批处理或存储的过程包含对用户定义函数的调用或执行动态SQL,例如使用 EXEC (字符串),则用户定义函数的已编译XML Showplan不包含在返回的表中通过sys.dm_exec_query_plan获取批处理或存储过程。相反,您必须单独调用sys.dm_exec_query_plan以获取与用户定义函数对应的计划句柄。

当即席查询使用简单或强制参数化时,query_plan列将包含仅语句文本,而不是实际查询计划。 若要返回查询计划,请调用sys.dm_exec_query_plan准备参数化查询的计划句柄。 您可以确定查询是否已参数化通过引用sql的列sys.syscacheobjects视图或文本列sys.dm_exec_sql_text动态管理视图。

注意:sys.dm_exec_query_plan返回的是实际执行计划。

2:使用SET SHOWPLAN_ALL ON 和SET SHOWPLAN_XML ON获取存储过程的执行计划。

如下所示,在AdventureWorks2014数据库中,查看存储过程[dbo].[uspGetEmployeeManagers] 的执行计划

SET SHOWPLAN_ALL ON

GO

SET FMTONLY ON

GO

EXEC dbo].[uspGetEmployeeManagers] 242;

GO

SET FMTONLY OFF

GO

SET SHOWPLAN_ALL OFF

GO

SET SHOWPLAN_ALL ON

GO

EXEC [dbo].[uspGetEmployeeManagers] 242;

GO

SET SHOWPLAN_ALL OFF;

GO

SET SHOWPLAN_XML ON

GO

EXEC [dbo].[uspGetEmployeeManagers] 242;

GO

SET SHOWPLAN_XML OFF;

GO

这种方式获取的是存储过程的预估执行计划,并不是实际执行计划,而且这种方式不会执行存储过程。另外,如果存储过程中存在临时表,那么就会出错。出现类似下面这样的错误:

Msg 208, Level 16, State 0, Procedure IntegrityTesting, Line 57

Invalid object name '#DbToCheck'.

3:SSMS中选中执行存储过程的脚本后,使用快捷方式CTRL+L或勾选“Display Estimated Execution Plan”获取存储过程的执行计划。

这种方式其实只是方式2(命令方式)的图形界面操作而已。所以此处不做展开

 

4:SSMS中使用快捷方式CTRL+M或勾选“Include Actual Execution plan”后,执行存储过程后,就能看到实际执行计划

这种方式看到的是存储过程的实际执行计划,但是很多时候,这种方式适用于开发、测试环境,但是我们在生产环境诊断问题时,这种方式并不适用。因为有时候存储过程会有DML操作,会修改数据。

5:SET STATISTICS XML 或SET STATISTICS PROFILE查看实际执行计划。

SET STATISTICS XML ON

GO

EXEC [dbo].[uspGetEmployeeManagers] 242;

GO

SET STATISTICS XML OFF;

GO

SET STATISTICS PROFILE ON

GO

EXEC [dbo].[uspGetEmployeeManagers] 242;

GO

SET STATISTICS PROFILE OFF;

GO

这个方式会执行存储过程。所以弊端上面也说过,有时候生产环境并不适用。

7:适用SQL Server Profile跟踪获取存储过程的实际执行计划

1: 打开SQL Server Profiler工具

2:在文件菜单,选择开启一个新的跟踪,跟踪对应的数据库服务器

3:“Events Section”选项卡中,"Use the template“选择“blank”

4: 然后在Performance节点,勾选相关字段。“Column Filter"设置过滤选项。例如只跟踪捕获某个数据库的某个存储过程的执行计划。如下例子所示

总结:我们可以使用上面方式获取存储过程的预估执行计划或实际执行计划。各有利弊,主要看使用场景,如果在生产环境诊断问题,优化SQL时,建议使用方式1 或 SQL Server Profiler, 因为有时候预估的执行计划,可能跟实际执行计划有出入,以及参数嗅探问题等等。

参考资料:

 

https://docs.microsoft.com/zh-cn/sql/relational-databases/system-dynamic-management-views/sys-dm-exec-query-plan-transact-sql?view=sql-server-2017

SQL Server如何查看存储过程的执行计划的更多相关文章

  1. SQL Server INSET/UPDATE/DELETE的执行计划

    DML操作符包括增删改查等操作方式. insert into Person.Address (AddressLine1, AddressLine2, City, StateProvinceID, Po ...

  2. SQL Server 性能调优 之执行计划(Execution Plan)调优

    SQL Server 存在三种 Join 策略:Hash Join,Merge Join,Nested Loop Join. Hash Join:用来处理没有排过序/没有索引的数据,它在内存中把 Jo ...

  3. 理解性能的奥秘——应用程序中慢,SSMS中快(2)——SQL Server如何编译存储过程

    本文属于<理解性能的奥秘--应用程序中慢,SSMS中快>系列 接上文:理解性能的奥秘--应用程序中慢,SSMS中快(1)--简介 本文介绍SQL Server如何编译存储过程并使用计划缓存 ...

  4. sql server内置存储过程、查看系统信息

    1.检索关键字:sql server内置存储过程,sql server查看系统信息 2.查看磁盘空间:EXEC master.dbo.xp_fixeddrives , --查看各个数据库所在磁盘情况S ...

  5. TFDStoredProc执行sql server的部分存储过程报错,有的是好的。

    TFDStoredProc执行sql server的部分存储过程报错,有的是好的. Invalid character value for cast specification 暂时无解.用fdque ...

  6. PL/SQL Developer 查看查询的执行计划

    https://zhuanlan.zhihu.com/p/65771352 通过 PL/SQL Developer 查看查询的执行计划 1 什么是执行计划 执行计划是一条查询语句在 Oracle 中的 ...

  7. Sql Server系列:存储过程

    1 存储过程简介 存储过程是使用T-SQL代码编写的代码段.在存储过程中,可以声明变量.执行条件判断语句等其他编程功能.在MS SQL Server 2012中存储过程主要分三类:系统存储过程.自定义 ...

  8. Sql Server 常用系统存储过程大全

    -- 来源于网络 -- 更详细的介结参考联机帮助文档 xp_cmdshell --*执行DOS各种命令,结果以文本行返回. xp_fixeddrives --*查询各磁盘/分区可用空间 xp_logi ...

  9. SQL点滴27—性能分析之执行计划

    原文:SQL点滴27-性能分析之执行计划 一直想找一些关于SQL语句性能调试的权威参考,但是有参考未必就能够做好调试的工作.我深信实践中得到的经验是最珍贵的,书本知识只是一个引导.本篇来源于<I ...

随机推荐

  1. C语言笔记 02_基本语法&数据类型&变量

    基本语法 令牌 C 程序由各种令牌组成,令牌可以是关键字.标识符.常量.字符串值,或者是一个符号.例如,下面的 C 语句包括五个令牌: printf("Hello, World! \n&qu ...

  2. ABAP里SELECT的用法汇总(转)

    通常使用Open SQL的数据查询语句SELECT将数据库条目选择到内存.一.SELECT语句:1)SELECT用于确定读取数据表中的哪些字段:2)FROM子句用于确定从哪些内表或者视图中读取数据:3 ...

  3. 从0使用Ruby on Rails打造企业级RESTful API项目实战之我的云音乐

    本节对我们项目实现的功能和知识点做一个简单的介绍,因为是RESTful API项目,所以对于后端来说基本上没有什么UI界面可展示,那我们就在关键的点,使用客户端(Android)实现的效果图. 课程简 ...

  4. Python基础语法-List

    列表的操作方法 列表中存放的数据是可以进行修改的,比如"增"."删"."改"" 添加元素("增" append ...

  5. 人体分析Demo-百度API

    本示例是采用Delphi 7 调用百度人体分析API:首先说明一下,怎么创建测试应用. 1.  登录百度云官网 https://cloud.baidu.com/ 当然需要一个百度账号 2.  进入管理 ...

  6. js对象数组中的某属性值 拼接成字符串

    js对象数组中的某属性值 拼接成字符串 var objs=[ {id:1,name:'张三'}, {id:2,name:'李四'}, {id:3,name:'王五'}, {id:4,name:'赵六' ...

  7. Python语法速查: 12. 文件与输入输出

    返回目录 (1)文件基本操作 ● 文件常用操作 内置函数或方法 描述 open(name [,mode [,buffering]]) 内置函数.用来打开文件,返回一个文件对象(file对象).详见下述 ...

  8. 粗糙集理论(Rough Set Theory)

    粗糙集理论(Rough Set Theory) 一种数据分析处理理论. <粗糙集—关于数据推理的理论>. 数据挖掘(Data Mining)和知识发现(KDD). 集合近似定义的基本思想及 ...

  9. ACM课内练习_1

    题意很简单就是一个一个素数因子只有2,3,5,7的整数,让你求它的约束的个数(暴力会超时),一开始的思路是先计算2,3,5,7这四个素数因子的个数,求出来之后想了很久没有想出他们个数和约束个数之间的数 ...

  10. StaticList

    到目前为止,我们还无法创建一个顺序存储结构线性表的对象出来,为什么呢?顺序存储空间究竟是什么,顺序存储空间的大小是多少? StaticList设计要点——类模板 使用原生数组作为顺序存储空间 使用模板 ...