如何编写高性能SQL存储过程循环_巧用集合代替游标操作

游标在SQL Server中性能极差,因其逐行处理、频繁触发查询计划和锁申请,应优先用CTE+ROW_NUMBER()分批处理或临时表+WHILE替代;仅极少数场景(如动态邮件发送)才需谨慎使用FAST_FORWARD只读游标。为什么游标在SQL Server里慢得像爬虫因为游标是逐行取、逐行处理,每次 fetch 都触发一次查询计划执行和锁资源申请,CPU 和 I/O 压力直接翻倍。尤其当 SELECT 返回上万行时,DECLARE CURSOR + FETCH NEXT 组合几乎必然成为性能瓶颈。真正该问的不是"怎么优化游标",而是"能不能根本不用游标"。答案通常是:能,而且必须换。游标默认是 KEYSET 或 STATIC 类型,内存开销大,且不支持并行执行WHILE 循环配合 TOP 1 + ORDER BY 模拟游标?更糟------它反复扫描表,没有索引友好性哪怕加了 WITH (NOLOCK),也只缓解读阻塞,不解决逻辑层低效用CTE + ROW_NUMBER()批量切片替代单行游标核心思路:把"一行一行处理"变成"一批一批处理",用集合运算一次生成所有需要的数据上下文,再通过窗口函数打序号,最后用 WHERE rn BETWEEN @start AND @end 分页驱动业务逻辑。适用于需按顺序处理但无强事务依赖的场景,比如日志归档、状态批量更新、中间表填充。先用 ROW_NUMBER() OVER (ORDER BY id) 给目标数据编号,别用 NEWID() ------ 它让排序不可控,且无法复用索引分片大小建议设为 5000~10000 行,太小则循环次数多;太大则单次事务日志暴涨,易触发 LOG FULL务必在 ORDER BY 字段建索引,否则 ROW_NUMBER() 会强制 SORT,内存消耗陡增WITH numbered AS ( SELECT *, ROW_NUMBER() OVER (ORDER BY created_time) AS rn FROM orders WHERE status = 'pending')UPDATE o SET status = 'processed'FROM orders oINNER JOIN numbered n ON o.id = n.idWHERE n.rn BETWEEN @offset + 1 AND @offset + @batch_size;临时表 + WHILE 循环的可控批量方案当业务逻辑复杂到无法全写进一个 UPDATE 或 INSERT ... SELECT 时,用临时表暂存主键集,再用 WHILE 控制批次,比游标轻量得多,且每批可独立提交事务。 arXiv Xplorer ArXiv 语义搜索引擎,帮您快速轻松的查找,保存和下载arXiv文章。

相关推荐
兵慌码乱4 小时前
基于 MediaPipe 与 PySide2 的手势交互音乐控制系统实现:轻量化视觉交互全流程解析
python·opencv·计算机视觉·人机交互·手势识别·mediapipe·pyside2
luckdewei7 小时前
FastAPI 资产管理系统实战:复杂 ORM 关联、Alembic 迁移与 N+1 查询优化
python
aqi0013 小时前
15天学会AI应用开发(八)使用向量数据库实现RAG功能
人工智能·python·大模型·ai编程·ai应用
Csvn14 小时前
`functools.lru_cache` —— 一行代码搞定缓存加速
后端·python
金銀銅鐵1 天前
[Python] 从《千字文》中随机挑选汉字
后端·python
cup111 天前
[技术复盘] Windows Python 打包实战:Nuitka 环境踩坑总结与 CI 自动化构建全指南
python·ai·环境变量·ci·nuitka·skill
aqi002 天前
15天学会AI应用开发(七)有了大模型为什么还要引入RAG
人工智能·python·大模型·ai编程·ai应用
金銀銅鐵2 天前
用 Python 实现 Take-Away 游戏
python·游戏