SQLServer如何监控阻塞会话

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

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 命令。

  • 请仔细测试程序以确保其符合预期行为,尤其是在生产环境中。

  • 日志文件的路径和权限需要根据实际情况进行配置。

文章转载自:Eric zhou

原文链接:https://www.cnblogs.com/tianqing/p/18217020

体验地址:引迈 - JNPF快速开发平台_低代码开发平台_零代码开发平台_流程设计器_表单引擎_工作流引擎_软件架构

相关推荐

  1. SQLServer如何监控阻塞

    2024-06-07 09:20:05       35 阅读
  2. GIN框架_

    2024-06-07 09:20:05       31 阅读
  3. oracle杀外部连接

    2024-06-07 09:20:05       41 阅读
  4. 【架构五】、事务、进程

    2024-06-07 09:20:05       41 阅读
  5. 达梦数据库kill

    2024-06-07 09:20:05       22 阅读

最近更新

  1. docker php8.1+nginx base 镜像 dockerfile 配置

    2024-06-07 09:20:05       98 阅读
  2. Could not load dynamic library ‘cudart64_100.dll‘

    2024-06-07 09:20:05       106 阅读
  3. 在Django里面运行非项目文件

    2024-06-07 09:20:05       87 阅读
  4. Python语言-面向对象

    2024-06-07 09:20:05       96 阅读

热门阅读

  1. Nginx R31 doc-09-Serving Static Content 静态内容

    2024-06-07 09:20:05       26 阅读
  2. Spring 怎么解决循环依赖的问题?

    2024-06-07 09:20:05       29 阅读
  3. Oracle 重置system用户的密码

    2024-06-07 09:20:05       29 阅读
  4. 9、架构-从类库到服务之服务发现

    2024-06-07 09:20:05       28 阅读
  5. 英语阅读文章

    2024-06-07 09:20:05       21 阅读
  6. 前端框架之 MVVM

    2024-06-07 09:20:05       28 阅读
  7. Docker基本架构概览-1

    2024-06-07 09:20:05       27 阅读
  8. Oracle数据库面试题-6

    2024-06-07 09:20:05       20 阅读
  9. Alembic维护python的数据库表结构

    2024-06-07 09:20:05       29 阅读
  10. 实现嵌入式软件定时、超时机制的常用方式

    2024-06-07 09:20:05       33 阅读
  11. 【深度学习基础】池化层

    2024-06-07 09:20:05       23 阅读
  12. 如何编辑 sudoers 文件

    2024-06-07 09:20:05       22 阅读