SQL Server存储过程本身不管理连接所谓“连接泄露”实为客户端如C#未正确释放SqlConnectionTRY...CATCH中的CLOSE/DEALLOCATE仅作用于游标或临时表无法关闭客户端连接正确做法是始终用using包裹SqlConnection确保Dispose调用。SQL Server 存储过程中连接没关异常后直接挂了存储过程本身不管理连接——它运行在已建立的会话里所谓“连接泄露”其实是调用方比如 C# 的 SqlConnection没正确释放但存储过程里的错误处理不当会让这个问题更难察觉。关键不是在存储过程里“关连接”而是确保调用链上每个 SqlConnection 都走完 Dispose() 或 using 块。为什么 TRY...CATCH 里写 CLOSE 和 DEALLOCATE 不起作用SQL Server 的 CLOSE 和 DEALLOCATE 只针对游标或临时表句柄对客户端发起的连接完全无效。连接生命周期由客户端控制T-SQL 层面无法主动断开外部连接。常见误解是以为在存储过程末尾加个 RETURN 就能“释放资源”其实只要客户端没调用 Close() 或让连接超出作用域连接就一直卡在 sleeping 状态。检查当前 sleeping 连接SELECT session_id, status, last_request_end_time FROM sys.dm_exec_sessions WHERE status sleepingTRY...CATCH 在存储过程中只捕获执行期错误不拦截网络中断、超时、客户端崩溃等场景即使存储过程内部报错并退出只要客户端没关闭连接连接仍保留在连接池中等待复用C# 调用时最常见的三个漏关连接的写法问题几乎全出在客户端代码。下面这三种写法看似正常但任意一个都可能导致连接堆积用 SqlCommand 直接执行但没包在 using 里也忘了手动调 connection.Close()在 try 块里打开连接但在 catch 或 finally 中漏掉 connection.Dispose()用了异步方法如 ExecuteReaderAsync但没 await 完就提前 return导致连接对象被 GC 延迟回收尤其在高并发下明显正确姿势只有一种using (var conn new SqlConnection(connStr)) { ... }哪怕里面只调一个存储过程也必须包住整个连接生命周期。 AI智研社 AI智研社是一个专注于人工智能领域的综合性平台