游标在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文章。
相关推荐
zore_c2 小时前
【C++】C++类和对象实现日期类项目——时间计算器!!!云边有个稻草人2 小时前
运营每次改数据都要等排期?我用飞牛NAS搭了个在线表格数据库,非技术也能自己管了m0_684501982 小时前
MySQL搭建主从后如何校验数据一致性_使用pt-table-sync修复差异草莓熊Lotso2 小时前
Linux 线程同步与互斥(二):线程同步从条件变量到生产者消费者模型全解,原理 + 源码彻底吃透孟意昶2 小时前
Doris专题28-聚合多维分析qq_206901392 小时前
SQL中如何处理多维数据的查询:复合索引与SELECT编写weixin_586061462 小时前
硬盘空间不足怎么装HTML工具_精简安装与外接存储方案【说明】m0_678485452 小时前
Redis如何在应用启动时预热缓存数据23471021274 小时前
4.18 学习笔记