在SQL Server数据库的日常运维和性能优化工作中,存储过程的执行频率是衡量业务负载分布、定位性能瓶颈的核心指标之一。高频率执行的存储过程如果存在性能问题,会对数据库整体响应速度产生明显影响,因此准确统计存储过程的执行频率是运维人员的必备技能。基于系统视图的统计分析方式无需修改数据库配置,也不会对业务运行产生额外负担,是最高效的分析途径。

核心系统视图介绍
SQL Server提供了多个系统视图用于存储过程的运行统计信息,其中最常用的是sys.dm_exec_procedure_stats,该视图会缓存所有已执行存储过程的聚合性能数据,数据会在SQL Server实例重启或者存储过程计划被清除时重置。
该视图的核心字段含义如下:
- database_id:存储过程所属的数据库ID,可关联
sys.databases获取数据库名称 - object_id:存储过程的对象ID,可关联
sys.objects获取存储过程名称 - execution_count:存储过程自上次编译以来的总执行次数
- total_elapsed_time:存储过程累计执行总耗时,单位为微秒
- last_execution_time:存储过程最后一次执行的时间
基础执行频率查询语句
通过关联sys.dm_exec_procedure_stats和sys.objects、sys.databases,可以获取所有存储过程的执行次数,按执行频率从高到低排序:
-- 查询所有存储过程执行频率,按执行次数降序排序
SELECT
DB_NAME(ps.database_id) AS 数据库名称,
OBJECT_NAME(ps.object_id, ps.database_id) AS 存储过程名称,
ps.execution_count AS 总执行次数,
ps.last_execution_time AS 最后执行时间,
ps.total_elapsed_time / 1000 AS 累计耗时_毫秒,
ps.total_elapsed_time / 1000 / ps.execution_count AS 平均耗时_毫秒
FROM sys.dm_exec_procedure_stats ps
INNER JOIN sys.objects o
ON ps.object_id = o.object_id
AND ps.database_id = DB_ID()
WHERE o.type = 'P' -- 筛选存储过程类型
ORDER BY ps.execution_count DESC;
按时间段过滤的查询方法
如果需要统计特定时间段内的存储过程执行频率,可以结合last_execution_time字段进行过滤,例如查询最近24小时内执行的存储过程:
-- 查询最近24小时内执行的存储过程执行频率
SELECT
DB_NAME(ps.database_id) AS 数据库名称,
OBJECT_NAME(ps.object_id, ps.database_id) AS 存储过程名称,
ps.execution_count AS 总执行次数,
ps.last_execution_time AS 最后执行时间
FROM sys.dm_exec_procedure_stats ps
INNER JOIN sys.objects o
ON ps.object_id = o.object_id
AND ps.database_id = DB_ID()
WHERE o.type = 'P'
AND ps.last_execution_time >= DATEADD(HOUR, -24, GETDATE())
ORDER BY ps.execution_count DESC;
结果分析与注意事项
通过查询结果可以重点关注执行次数Top10的存储过程,如果这些存储过程的平均耗时较高,优先对其进行性能优化。需要注意以下几点:
- 系统视图的数据是实例级别的缓存数据,SQL Server重启、手动清除计划缓存、存储过程定义被修改后,统计数据会被重置
- 对于执行频率极低或者从未执行过的存储过程,该视图中不会存在对应记录,需要查询
sys.objects获取全量存储过程列表 - 如果数据库启用了查询存储功能,也可以结合查询存储的历史数据获取更长时间段的执行频率统计
扩展:导出统计结果到本地文件
如果需要将统计结果保存为本地文件,可以使用BCP命令导出查询结果,示例如下:
-- 先执行查询并将结果插入临时表
SELECT
DB_NAME(ps.database_id) AS 数据库名称,
OBJECT_NAME(ps.object_id, ps.database_id) AS 存储过程名称,
ps.execution_count AS 总执行次数,
ps.last_execution_time AS 最后执行时间
INTO #ProcExecStats
FROM sys.dm_exec_procedure_stats ps
INNER JOIN sys.objects o
ON ps.object_id = o.object_id
AND ps.database_id = DB_ID()
WHERE o.type = 'P';
-- BCP导出命令,在命令行执行
-- bcp "SELECT * FROM #ProcExecStats" queryout "D:proc_exec_stats.csv" -c -t, -T -S localhost