一、查询阻塞和被阻塞的会话

SELECT
r.session_id AS [Blocked Session ID],
r.blocking_session_id AS [Blocking Session ID],
r.wait_type,
r.wait_time,
r.wait_resource,
s1.program_name AS [Blocked Program Name],
s1.login_name AS [Blocked Login],
s2.program_name AS [Blocking Program Name],
s2.login_name AS [Blocking Login],
r.text AS [SQL Text]
FROM sys.dm_exec_requests AS r
LEFT JOIN sys.dm_exec_sessions AS s1 ON r.session_id = s1.session_id
LEFT JOIN sys.dm_exec_sessions AS s2 ON r.blocking_session_id = s2.session_id
CROSS APPLY sys.dm_exec_sql_text(r.sql_handle) AS r
WHERE r.blocking_session_id <> 0;

二、找出阻塞的具体SQL

SELECT
r.session_id,
r.blocking_session_id,
t.text AS [SQL Text],
r.wait_type,
r.wait_time,
r.wait_resource
FROM sys.dm_exec_requests AS r
CROSS APPLY sys.dm_exec_sql_text(r.sql_handle) AS t
WHERE r.blocking_session_id <> 0;

三、编写C#程序,每隔10秒监控SQL Server数据库中的阻塞会话,定位出阻塞的根源会话并终止它们,同时记录日志。

using System;
using System.Data.SqlClient;
using System.IO;
using System.Timers; class Program
{
private static Timer timer;
private static string connectionString = "your_connection_string_here"; static void Main(string[] args)
{
timer = new Timer(10000); // 每10秒执行一次
timer.Elapsed += CheckForBlockingSessions;
timer.AutoReset = true;
timer.Enabled = true; Console.WriteLine("Press [Enter] to exit the program.");
Console.ReadLine();
} private static void CheckForBlockingSessions(object source, ElapsedEventArgs e)
{
try
{
using (SqlConnection connection = new SqlConnection(connectionString))
{
connection.Open(); string query = @"
SELECT
r.session_id AS BlockedSessionID,
r.blocking_session_id AS BlockingSessionID,
r.text AS SqlText
FROM sys.dm_exec_requests AS r
CROSS APPLY sys.dm_exec_sql_text(r.sql_handle) AS r
WHERE r.blocking_session_id <> 0;"; using (SqlCommand command = new SqlCommand(query, connection))
{
using (SqlDataReader reader = command.ExecuteReader())
{
while (reader.Read())
{
int blockedSessionId = reader.GetInt32(0);
int blockingSessionId = reader.GetInt32(1);
string sqlText = reader.GetString(2); LogBlockingSession(blockedSessionId, blockingSessionId, sqlText);
KillSession(blockingSessionId);
}
}
}
}
}
catch (Exception ex)
{
LogError(ex.Message);
}
} private static void KillSession(int sessionId)
{
try
{
using (SqlConnection connection = new SqlConnection(connectionString))
{
connection.Open();
string killQuery = $"KILL {sessionId};";
using (SqlCommand killCommand = new SqlCommand(killQuery, connection))
{
killCommand.ExecuteNonQuery();
LogKillSession(sessionId);
}
}
}
catch (Exception ex)
{
LogError($"Failed to kill session {sessionId}: {ex.Message}");
}
} private static void LogBlockingSession(int blockedSessionId, int blockingSessionId, string sqlText)
{
string logMessage = $"[{DateTime.Now}] Blocked Session ID: {blockedSessionId}, Blocking Session ID: {blockingSessionId}, SQL Text: {sqlText}";
File.AppendAllText("blocking_sessions.log", logMessage + Environment.NewLine);
Console.WriteLine(logMessage);
} private static void LogKillSession(int sessionId)
{
string logMessage = $"[{DateTime.Now}] Killed Session ID: {sessionId}";
File.AppendAllText("killed_sessions.log", logMessage + Environment.NewLine);
Console.WriteLine(logMessage);
} private static void LogError(string message)
{
string logMessage = $"[{DateTime.Now}] Error: {message}";
File.AppendAllText("errors.log", logMessage + Environment.NewLine);
Console.WriteLine(logMessage);
}
}

说明

  1. 连接字符串:替换 your_connection_string_here 为实际的数据库连接字符串。
  2. 定时器:使用 System.Timers.Timer 类设置每10秒执行一次检查。
  3. 检查阻塞会话:在 CheckForBlockingSessions 方法中,查询阻塞会话和根源会话的信息。
  4. 终止会话:在 KillSession 方法中,执行 KILL 命令来终止阻塞会话。
  5. 日志记录:日志记录包括阻塞会话的详细信息和终止会话的操作,以及错误信息。

注意事项

  • 运行此程序需要确保有足够的权限来访问数据库和执行 KILL 命令。
  • 请仔细测试程序以确保其符合预期行为,尤其是在生产环境中。
  • 日志文件的路径和权限需要根据实际情况进行配置。

周国庆

2024/5/28

SQLServer如何监控阻塞会话的更多相关文章

  1. sqlserver监控阻塞(死锁)具体情况

    公司sqlserver的监控系统主要是采用zabbix监控,但是zabbix的监控只能通过性能计数器给出报警,而无法给出具体的阻塞情况,比如阻塞会话.语句.时间等,所以需要配合sqlserver的一些 ...

  2. sqlserver监控阻塞(死锁)具体情况(转)

    公司sqlserver的监控系统主要是采用zabbix监控,但是zabbix的监控只能通过性能计数器给出报警,而无法给出具体的阻塞情况,比如阻塞会话.语句.时间等,所以需要配合sqlserver的一些 ...

  3. SQLSERVER性能监控级别步骤

    SQLSERVER性能监控级别步骤 下面先用一幅图描述一下有哪些步骤和顺序 1.识别瓶颈 识别瓶颈的原因包括多个方面,例如,资源不足,需要添加或升级硬件: 工作负荷在同类资源之间分布不均匀,例如,一个 ...

  4. 如何定位Oracle数据库被锁阻塞会话的根源

    首先再次明确下,数据库因为要同时保证数据的并发性和一致性,所以操作有锁等待是正常的. 只有那些长时间没有提交或回滚的事物,阻塞了其他业务正常操作,才是需要去定位处理的. 1.单实例环境 2.RAC环境 ...

  5. SQLServer数据库监控代码

    SQLServer数据库监控代码: creation_time, total_worker_time, last_worker_time, max_worker_time, min_worker_ti ...

  6. 现在的SQLSERVER数据库监控软件有哪些?

    现在的SQLSERVER数据库监控软件有哪些? 收集了一下当前SQLSERVER数据库监控软件,发现开源免费的真的是“没有” Questsoftware  Quest's spotlight(收费)  ...

  7. 【SQL Server DBA】日常巡检语句3:特定监控(阻塞、top语句、索引、作业)

    原文:[SQL Server DBA]日常巡检语句3:特定监控(阻塞.top语句.索引.作业) 1.查询阻塞信息.锁定了哪些资源 --1.查看阻塞信息 select spid,loginame,wai ...

  8. Oracle阻塞会话源头查找-单机和RAC环境

    在写 Oracle session相关数据字典(一)  这篇文章时,提到使用v$session视图的树形查询可以得到Oracle锁树,这样就便于我们找出阻塞会话的源头,但是仅仅可以在单机环境中使用.今 ...

  9. MySQL的共享锁阻塞会话案例浅析输入日志标题

        这是问题是一个网友遇到的问题:一个UPDATE语句产生的共享锁阻塞了其他会话的案例,对于这个案例,我进一步分析.总结和衍化了相关问题.下面分析如有不对的地方,敬请指正.下面是初始化环境和数据的 ...

  10. zabbix自定义监控(当会话登录超过三个就报警)

    安装过程在此省略. 1.agent端去修改配置文件 2.调用自定义内容 vim /etc/zabbix/zabbix_agentd.d/login.conf UserParameter=login-u ...

随机推荐

  1. CentOS8 / CentOS7 yum源最新修改搭建 2022.3.1

    Part I CentOS 8 源更新 ========================================== 2022年过完后,发现公司里面的所有服务器yum都不能用了,一直报错 按照 ...

  2. UML 哲学之道——概况篇[二]

    前言 简单介绍一下uml的概况篇. 正文 UML 概述: url 包括: 事物 关系 图 扩展机制 事物: 结构: 类.接口.构件.节点等等 行为:交互.状态等等 分组:包.子系统等等 注释:注释 关 ...

  3. redis 简单整理——哨兵原理[三十一]

    前言 简单介绍一下哨兵的原理. 正文 一套合理的监控机制是Sentinel节点判定节点不可达的重要保证,Redis Sentinel通过三个定时监控任务完成对各个节点发现和监控: 1)每隔10秒,每个 ...

  4. 数据结构实验代码分享 - 3 (哈夫曼树 / HuffmanTree)

    哈夫曼编码/ 译码系统(树应用) [问题描述] 任意给定一个仅由 26 个大写英文字母组成的字符序列,根据哈夫曼编码算法,求得每个字符的哈夫曼编码. 要求: 1)输入一个由 26 个英文字母组成的字符 ...

  5. drf——基于apiview写过滤、排序和分页

    基于APIView带过滤和排序 from rest_framework.views import APIView from .models import Book from .serializer i ...

  6. 技术人如何"结构化"高效思考?

    1 前言 你是否在解决问题时,常常觉得脑子很乱,很多零散的信息迸发出来,但就是无法形成解决方案. 之所以这样,原因就在于,我们的大脑处理不了太多零散而复杂的信息.人类大脑在处理信息的时候,有两个规律: ...

  7. 力扣1112(MySQL)-每位学生的最高成绩(中等)

    题目: 表:Enrollments (student_id, course_id) 是该表的主键. 问题编写一个 SQL 查询,查询每位学生获得的最高成绩和它所对应的科目,若科目成绩并列,取 cour ...

  8. 力扣242(java)-有效的字母异位词(简单)

    题目: 给定两个字符串 s 和 t ,编写一个函数来判断 t 是否是 s 的字母异位词. 注意:若 s 和 t 中每个字符出现的次数都相同,则称 s 和 t 互为字母异位词. 示例 1: 输入: s ...

  9. 冷热分离之OTS表格存储实战

    简介: 为什么要冷热分离由于2020疫情的原因,在线教育行业提前被大家所重视,钉钉教育已经服务超过21万所学校.700万教师和1.4亿学生用户,每天大量的教育数据产生.整体数据量:随着时间的积累,数据 ...

  10. 重磅发布:微服务引擎 MSE 专业版

    简介: 性能提升 10 倍,更高的 SLA 保障,新用户限时抢购 8 折资源包. 微服务引擎 MSE 专业版发布,支持 Nacos 2.0 ,相比基础版,专业版具有更高的 SLA 保障,性能提升十倍, ...