如何编写高性能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文章。

相关推荐
雨辰AI19 小时前
SpringBoot3 整合达梦 DM9 超详细入门实战|从零搭建可直接上线
数据库·微服务·架构·政务
原来是猿19 小时前
博客系统自动化测试实战总结
python
我是一颗柠檬19 小时前
【MySQL全面教学】MySQL性能优化实战Day13(2026年)
数据库·后端·sql·mysql·性能优化·database
AI人工智能+电脑小能手19 小时前
【大白话说Java面试题 第84题】【Mysql篇】第14题:为什么用 InnoDB 存储引擎的表建议用整型的自增主键?
java·开发语言·数据库·mysql·面试
小江的记录本19 小时前
【JVM虚拟机】JVM调优:常用JVM参数、调优核心指标、OOM排查、GC日志分析、Arthas工具使用(附《思维导图》+《面试高频考点清单》)
java·jvm·spring boot·后端·python·spring·面试
大数据魔法师19 小时前
Streamlit(十三)- API 参考文档(六)- 媒体展示组件
python·web
张彦峰ZYF19 小时前
检索增强生成(RAG)系统的基础:全面深入矢量数据库
数据库·大模型·rag
爱写代码的倒霉蛋20 小时前
Hello-Agents的第一个练习-5分钟实现一个智能体(实现详解)
python
金銀銅鐵20 小时前
[Java] 用图形化界面演示 iadd, isub, iconst_<i> 指令的效果
java·后端·python
春日见20 小时前
五分钟入门 强化学习---DQN(Deep Q Net)算法与实现
人工智能·python·深度学习·算法·microsoft·机器学习