0

0

如何提升SQL存储过程执行稳定性_添加超时检查与重试

P粉602998670

P粉602998670

发布时间:2026-03-19 11:48:43

|

942人浏览过

|

来源于php中文网

原创

SET LOCK_TIMEOUT不能直接救急,但它是SQL Server中唯一能在语句级控制锁等待时长的原生机制,需配合TRY...CATCH与手动重试逻辑使用,且须区分应用层超时与锁超时。

如何提升sql存储过程执行稳定性_添加超时检查与重试

存储过程执行卡死时,SET LOCK_TIMEOUT 能不能救急

不能直接“救急”,但它是唯一能在语句级控制阻塞等待时长的原生机制。SQL Server 不支持在存储过程中全局设置查询超时(那是应用层或连接字符串的事),SET LOCK_TIMEOUT 只影响后续的锁等待行为,不终止已运行的计算逻辑。

常见错误现象:sp_who2 显示状态为 sleeping 但会话长期不返回,实际是卡在锁等待里;或者日志里反复出现 Timeout expired. The timeout period elapsed prior to completion of the operation,但源头不是网络或连接池——而是某条 UPDATE 被另一事务锁住太久。

  • SET LOCK_TIMEOUT 5000 表示最多等 5 秒,超时抛出错误号 1205(注意:不是死锁错误号,死锁是 1204
  • 必须在存储过程开头显式设置,作用域仅限当前会话,不会继承自调用方
  • SELECT 有效,但对 INSERT/UPDATE/DELETE 中的锁争用更关键;对 CPU 密集型操作(如大表排序、递归 CTE)完全无效
  • 值设为 -1 是默认行为(无限等待),设为 0 则完全不等待,冲突立即失败

重试逻辑必须由存储过程自己实现,TRY...CATCH 是基础但不够

SQL Server 没有内置重试关键字,所有重试都要靠 WHILE + TRY...CATCH 手写循环。只靠 CATCH 捕获错误并 RETURN,等于放弃重试机会。

使用场景:上游系统无法控制重试(比如 legacy 客户端只发一次调用),或业务要求“尽力而为”(如日志落库、状态同步类操作)。

  • 必须在 CATCH 块中检查 ERROR_NUMBER(),只对 1205(死锁)、1222(锁超时)、1204(死锁牺牲品)这类可重试错误才执行重试
  • 重试前加 WAITFOR DELAY '00:00:00.1',避免瞬间重试加剧锁竞争;延迟建议从 100ms 起步,逐次倍增(最多 3 次)
  • 用局部变量记录重试次数,防止无限循环;超过阈值应 THROW 原始错误,让调用方兜底
  • 注意事务嵌套:如果外层已有事务,ROLLBACK 会回滚整个事务,重试前需确保事务状态可恢复(通常建议重试逻辑独占事务)

应用层超时和存储过程内超时要分清楚,别互相覆盖

连接字符串里的 Connection Timeout=30 控制的是连上 SQL Server 的耗时,和存储过程执行无关;CommandTimeout(如 ADO.NET 的 SqlCommand.CommandTimeout)才是真正的执行超时开关,它触发的是客户端强制断开,SQL Server 端会收到 ERROR 0 并回滚当前批处理。

WisPaper
WisPaper

复旦大学研发的AI学术搜索工具,5分钟内筛选1000篇论文

下载

容易踩的坑:在存储过程中设了 SET LOCK_TIMEOUT 3000,但应用层 CommandTimeout 设成 10 秒,结果锁等 3 秒失败后重试两次,每次又卡 3 秒,第 3 次还没跑完就被客户端砍掉——此时你看到的错误是 Execution Timeout Expired,而不是你预设的锁超时错误。

  • 存储过程内超时(LOCK_TIMEOUT)解决的是“等不等得及”,应用层超时解决的是“给不给足够时间”
  • 两者应配合:应用层超时 ≥ 单次最大可能耗时 × 重试次数 + 预留缓冲(例如单次最长 8 秒,重试 3 次,设 CommandTimeout=30
  • 不要在存储过程中用 WAITFOR 模拟超时,它不释放资源,还拖慢整个会话

真正稳定的重试需要识别“可重试”和“不可重试”错误类型

不是所有报错都能重试。盲目重试 CONVERT 失败、主键冲突、权限不足这类错误,只会让问题更快暴露为高频告警。

性能影响很实际:每次重试都重新走一遍执行计划、重新申请锁、重新读数据页。若底层是高并发更新热点行,重试反而放大锁争用。

  • 安全重试的错误号: 1204(死锁牺牲品)、1205(死锁)、1222(锁超时)、1221(锁升级失败)
  • 禁止重试的错误号: 2627(唯一键冲突)、547(外键约束)、245(类型转换失败)、208(对象名无效)
  • 建议把错误分类逻辑封装成标量函数,例如 dbo.IsRetryableError(@errnum),避免每个存储过程重复写判断
  • 重试日志必须记清楚:原始错误号、重试次数、最终是否成功——否则出问题时根本分不清是第几次失败

最麻烦的其实是那些没报错但“不动”的情况:比如某个 UPDATE 匹配了 100 万行,但 WHERE 条件没走索引,实际在扫全表。这种时候 LOCK_TIMEOUT 和重试都无能为力,得靠执行计划分析和索引优化——那已经是另一层稳定性问题了。

相关标签:

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
sqlserver和mysql区别
sqlserver和mysql区别

SQL Server和MySQL是两种广泛使用的关系型数据库管理系统。它们具有相似的功能和用途,但在某些方面存在一些显著的区别。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

543

2023.08.11

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

391

2023.06.29

如何删除数据库
如何删除数据库

删除数据库是指在MySQL中完全移除一个数据库及其所包含的所有数据和结构,作用包括:1、释放存储空间;2、确保数据的安全性;3、提高数据库的整体性能,加速查询和操作的执行速度。尽管删除数据库具有一些好处,但在执行任何删除操作之前,务必谨慎操作,并备份重要的数据。删除数据库将永久性地删除所有相关数据和结构,无法回滚。

2113

2023.08.14

vb怎么连接数据库
vb怎么连接数据库

在VB中,连接数据库通常使用ADO(ActiveX 数据对象)或 DAO(Data Access Objects)这两个技术来实现:1、引入ADO库;2、创建ADO连接对象;3、配置连接字符串;4、打开连接;5、执行SQL语句;6、处理查询结果;7、关闭连接即可。

359

2023.08.31

MySQL恢复数据库
MySQL恢复数据库

MySQL恢复数据库的方法有使用物理备份恢复、使用逻辑备份恢复、使用二进制日志恢复和使用数据库复制进行恢复等。本专题为大家提供MySQL数据库相关的文章、下载、课程内容,供大家免费下载体验。

259

2023.09.05

vb中怎么连接access数据库
vb中怎么连接access数据库

vb中连接access数据库的步骤包括引用必要的命名空间、创建连接字符串、创建连接对象、打开连接、执行SQL语句和关闭连接。本专题为大家提供连接access数据库相关的文章、下载、课程内容,供大家免费下载体验。

329

2023.10.09

数据库对象名无效怎么解决
数据库对象名无效怎么解决

数据库对象名无效解决办法:1、检查使用的对象名是否正确,确保没有拼写错误;2、检查数据库中是否已存在具有相同名称的对象,如果是,请更改对象名为一个不同的名称,然后重新创建;3、确保在连接数据库时使用了正确的用户名、密码和数据库名称;4、尝试重启数据库服务,然后再次尝试创建或使用对象;5、尝试更新驱动程序,然后再次尝试创建或使用对象。

420

2023.10.16

vb连接access数据库的方法
vb连接access数据库的方法

vb连接access数据库方法:1、使用ADO连接,首先导入System.Data.OleDb模块,然后定义一个连接字符串,接着创建一个OleDbConnection对象并使用Open() 方法打开连接;2、使用DAO连接,首先导入 Microsoft.Jet.OLEDB模块,然后定义一个连接字符串,接着创建一个JetConnection对象并使用Open()方法打开连接即可。

482

2023.10.16

bootstrap安装教程
bootstrap安装教程

本专题整合了bootstrap安装相关教程,阅读专题下面的文章了解更多详细操作教程。

22

2026.03.18

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号