sql server怎样用扩展事件捕获带变量值的慢sql

sqlserver查询当前执行的sql时,经常看到带变量如@PO,@P1的sql,没有具体的值,有的sql甚至有几十个变量,优化的时候需要一个个手动代入,很是不方便。

用扩展事件可以捕获带变量值的慢sql。

当cpu高的时候,可以创建并启动个扩展事件来捕获慢sql。

已验证在always on集群从库上创建扩展事件,不影响主从数据同步。

本文以捕获超过一秒的慢sql为例。

一 创建扩展事件

IF EXISTS(SELECT * FROM sys.server_event_sessions WHERE name = 'slow_query')  

    DROP EVENT SESSION slow_query ON SERVER   

GO  

--  创建事件会话  

CREATE EVENT SESSION [slow_query] ON SERVER 

ADD EVENT sqlserver.rpc_completed(

    ACTION(sqlos.task_time,sqlserver.database_name,sqlserver.nt_username,sqlserver.sql_text,sqlserver.transaction_id,sqlserver.username,sqlserver.client_hostname,sqlserver.client_app_name)

    WHERE  [duration]>=1000000                    -- 消耗秒数超 1 秒      

    ),

ADD EVENT sqlserver.sql_batch_completed(

    ACTION(sqlos.task_time,sqlserver.database_name,sqlserver.nt_username,sqlserver.sql_text,sqlserver.transaction_id,sqlserver.username,sqlserver.client_hostname,sqlserver.client_app_name)

    WHERE   [duration]>=1000000                    -- 消耗秒数超 1 秒  

      )

ADD TARGET package0.event_file(SET filename=N'D:\test\slow_query.xel',max_file_size=(1000))    --路径、文件MB, 自行修改

WITH (STARTUP_STATE=ON)

GO

--  启用(停止)事件会话(START / STOP)  

ALTER EVENT SESSION slow_query ON SERVER STATE=START  

GO

二 模拟慢sql-执行一个大于1秒的慢sql

select * from test2 where id>15000000

三 查看监测到的数据

有两种方式可以查询。

用SSMS可以看到每条慢sql的详细记录,但如果sql较多,可以用sql筛选想查的数据,比较灵活。

3.1 用SSMS查

--备注:扩展事件创建之后执行的慢sql也能捕获到。

3.2 用sql查

查看下生成的xel文件,手动改下下面的xel文件名

3.2.1 查看具体一个个慢sql记录

with d as (

SELECT CONVERT(XML,event_data) AS data

from sys.Fn_xe_file_target_read_file(N'D:\test\slow_query_0_133763843782130000.xel',NULL,NULL,NULL)

)

select

dateadd(hour,8,data.value('(/event/@timestamp)[1]','datetime')) as record_time, --获取最上方标题行的内容

data.value('(/event/data[@name="cpu_time"]/value)[1]','int')/1000 as 'cpu_time(ms)',--获得 event=>data name=cpu_time 的 value

data.value('(/event/data[@name="duration"]/value)[1]','int')/1000 as 'exec_time(ms)',--获得 event=>data name=duration 的 value

--data.value('(/event/data[@name="batch_text"]/value)[1]','nvarchar(max)') as 'batch_text',--获得 event=>data name=batch_text 的 text

data.value('(/event/action[@name="sql_text"]/value)[1]','nvarchar(4000)') as 'current_param_sql',--获得 event=>action name=sql_text 的 value
data.value('(/event/data[@name="statement"]/value)[1]','nvarchar(4000)') as 'current_value_sql',--获得 event=>data name=statement的 value

--data.value('(/event/action[@name="task_time"]/value)[1]','int')/1000 as 'task_time(ms)',--获得 event=>action name=task_time 的 value

data.value('(/event/action[@name="database_name"]/value)[1]','nvarchar(400)') as 'database_name',--获得 event=>action name=database_name 的 value

data.value('(/event/action[@name="transaction_id"]/value)[1]','nvarchar(400)') as 'transaction_id',--获得 event=>action name=transaction_id 的 value

data.value('(/event/action[@name="username"]/value)[1]','nvarchar(400)') as 'username',--获得 event=>action name=username 的 value

data.value('(/event/action[@name="nt_username"]/value)[1]','nvarchar(400)') as 'nt_username',--获得 event=>action name=nt_username 的 value

data.value('(/event/action[@name="client_hostname"]/value)[1]','nvarchar(400)') as 'client_hostname',--获得 event=>action name=client_hostname 的 value

data.value('(/event/action[@name="client_app_name"]/value)[1]','nvarchar(400)') as 'client_app_name',--获得 event=>action name=client_app_name 的 value

data.value('(/event/@name)[1]','nvarchar(128)') as operation_name, --获取最上方标题行的内容

data.value('(/event/data[@name="physical_reads"]/value)[1]','int') as 'physical_reads',--获得 event=>data name=physical_reads 的 value

data.value('(/event/data[@name="logical_reads"]/value)[1]','int') as 'logical_reads',--获得 event=>data name=logical_reads 的 value

data.value('(/event/data[@name="writes"]/value)[1]','int') as 'writes',--获得 event=>data name=writes 的 value

data.value('(/event/data[@name="row_count"]/value)[1]','int') as 'row_count',--获得 event=>data name=row_count 的 value

data.value('(/event/data[@name="result"]/value)[1]','int') as 'result_flag',--获得 event=>data name=result 的 value

data.value('(/event/data[@name="result"]/text)[1]','nvarchar(128)') as 'result_desc'--获得 event=>data name=result 的 text

from d

order by "cpu_time(ms)" desc

3.2.3 查看汇总版的慢sql(某个sql执行次数,平均执行时长,总体执行时长等信息)

with d as (
 
SELECT CONVERT(XML,event_data) AS data
 
from sys.Fn_xe_file_target_read_file(N'D:\test\slow_query_0_133764079441250000.xel',NULL,NULL,NULL)
 
),
 e as
(
 
select 
 
dateadd(hour,8,data.value('(/event/@timestamp)[1]','datetime')) as record_time, --获取最上方标题行的内容
 --
data.value('(/event/data[@name="cpu_time"]/value)[1]','int')/1000 as 'cpu_time_ms',--获得 event=>data name=cpu_time 的 value
 
data.value('(/event/data[@name="duration"]/value)[1]','int')/1000 as 'exec_time_ms',--获得 event=>data name=duration 的 value
 
--data.value('(/event/data[@name="batch_text"]/value)[1]','nvarchar(max)') as 'batch_text',--获得 event=>data name=batch_text 的 text
 
data.value('(/event/action[@name="sql_text"]/value)[1]','nvarchar(4000)') as 'current_param_sql',--获得 event=>action name=sql_text 的 value
data.value('(/event/data[@name="statement"]/value)[1]','nvarchar(4000)') as 'current_value_sql',--获得 event=>data name=sql_text 的 value
 
 
--data.value('(/event/action[@name="task_time"]/value)[1]','int')/1000 as 'task_time(ms)',--获得 event=>action name=task_time 的 value
 
data.value('(/event/action[@name="database_name"]/value)[1]','nvarchar(400)') as 'database_name',--获得 event=>action name=database_name 的 value
 
data.value('(/event/action[@name="transaction_id"]/value)[1]','nvarchar(400)') as 'transaction_id',--获得 event=>action name=transaction_id 的 value
 
data.value('(/event/action[@name="username"]/value)[1]','nvarchar(400)') as 'username',--获得 event=>action name=username 的 value
 
data.value('(/event/action[@name="nt_username"]/value)[1]','nvarchar(400)') as 'nt_username',--获得 event=>action name=nt_username 的 value
 
data.value('(/event/action[@name="client_hostname"]/value)[1]','nvarchar(400)') as 'client_hostname',--获得 event=>action name=client_hostname 的 value
 
data.value('(/event/action[@name="client_app_name"]/value)[1]','nvarchar(400)') as 'client_app_name',--获得 event=>action name=client_app_name 的 value
 
data.value('(/event/@name)[1]','nvarchar(128)') as operation_name, --获取最上方标题行的内容
 
data.value('(/event/data[@name="physical_reads"]/value)[1]','int') as 'physical_reads',--获得 event=>data name=physical_reads 的 value
 
data.value('(/event/data[@name="logical_reads"]/value)[1]','int') as 'logical_reads',--获得 event=>data name=logical_reads 的 value
 
data.value('(/event/data[@name="writes"]/value)[1]','int') as 'writes',--获得 event=>data name=writes 的 value
 
data.value('(/event/data[@name="row_count"]/value)[1]','int') as 'row_count',--获得 event=>data name=row_count 的 value
 
data.value('(/event/data[@name="result"]/value)[1]','int') as 'result_flag',--获得 event=>data name=result 的 value
 
data.value('(/event/data[@name="result"]/text)[1]','nvarchar(128)') as 'result_desc'--获得 event=>data name=result 的 text

from d 
)
select database_name,current_param_sql,max(current_value_sql) as current_value_sql,count(*) as cnt,sum(cpu_time_ms) as sum_cpu_time_ms,sum(exec_time_ms) as sum_exec_time_ms,avg(cpu_time_ms) as avg_cpu_time_ms,avg(exec_time_ms) as avg_exec_time_ms,max(username) as username,max(client_hostname) as client_hostname,max(client_app_name) as client_app_name,max(row_count) as max_row_count
from e 
group by current_param_sql,database_name
order by sum_cpu_time_ms desc

--本篇文章参考了

【监控笔记】【2.3】扩展事件------慢查询SQL(执行超过3S的SQL) - 郭大侠1 - 博客园

并做了些许改动。

相关推荐
明矾java20 分钟前
Mysql-SQL执行流程解析
数据库·sql·mysql
蓬莱道人37 分钟前
BenchmarkSQL使用教程
数据库
p@nd@1 小时前
Oracle筑基篇-调度算法-LRU的引入
数据库·oracle·操作系统·lru
来一杯龙舌兰1 小时前
【MongoDB】使用 MongoDB 存储日志、审批、MQ等数据的案例及优点
数据库·mongodb
技术路上的苦行僧1 小时前
分布式专题(8)之MongoDB存储原理&多文档事务详解
数据库·分布式·mongodb
孤独的履行者1 小时前
入门靶机:DC-1的渗透测试
数据库·python·网络安全
wy02_1 小时前
MySQL-MVCC(多版本并发控制)
数据库·mysql
NY63 小时前
mysql运维篇笔记——日志,主从复制,分库分表,读写分离
数据库·sql
潜洋3 小时前
Spring Boot 教程之三十六:实现身份验证
java·数据库·spring boot
科马3 小时前
【Redis】缓存
数据库·redis·spring·缓存