游标在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文章。
相关推荐
辞旧 lekkk3 小时前
【Qt】信号和槽2301_809204704 小时前
JavaScript中严格模式use-strict对引擎解析的辅助.txtzjy277775 小时前
mysql如何选择合适的索引类型_mysql索引设计实战Aaswk5 小时前
Java Lambda 表达式与流处理笨蛋不要掉眼泪5 小时前
Mysql架构揭秘:update语句的执行流程万邦科技Lafite5 小时前
京东item_get接口实战案例:实时商品价格监控全流程解析秋96 小时前
ruoyi项目更换为mysql9.7.0数据库Andya_net6 小时前
MySQL | MySQL 8.0 权限管理实践-精确赋予库、表只读等权限Cyber4K6 小时前
【Python专项】进阶语法-系统资源监控与数据采集(1)冷小鱼6 小时前
JVM 异常崩溃排查全指南:从 Core Dump 到根因定位