记录SQL Server中游标是如何使用

本文详细介绍了SQL游标的概念,包括其使用场景、静态游标与动态游标的区别,以及如何创建和使用游标处理逐行数据。特别提到游标可能导致的性能开销,提醒开发者合理运用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在日常的工作中,遇到了关于sql中游标的问题。在网上查找资料以后,特此记录一下。

首先说明一下游标的使用场景:

我们平常对数据库中的数据进行增删改时,都是一次性处理若干行符合指定条件的数据,而如果有个业务场景,要求我们对某些数据逐行进行判断并采取不同的方式处理,这时候一般的语句就满足不了了。

游标可以方便从一个结果集中进行循环遍历数据在进行操作。它的思想就是从结果集中,一个一个取出来处理.当然,由于它是逐一处理的方式,游标的最大弊端就是低效。

游标的类型:

游标分为两种,静态游标和动态游标,默认的都是动态游标

静态游标在打开时会将数据集存储在tempdb中,因此显示的数据与游标打开时的数据集保持一致,在游标打开以后对数据库的更新不会显示在游标中。

动态游标在打开后会反映对数据库的更改。所有UPDATE、INSERT 和 DELETE 操作都会显示在游标的结果集中,结果集中的行数据值、顺序和成员在每次提取时都会改变。

如何创建一个游标:

--创建一个游标
declare my_cursor cursor STATIC --my_cursor为游标的名称,随便起,STATIC加上就是静态游标,去掉就是动态游标
for     
select name,id from table      --这是游标my_cursor的值,这里随便发挥看业务场景
--打开游标
open my_cursor                  --打开游标
--变量
declare   @name varchar(150)   --声明变量  ‘declare’为声明变量 ‘@name’为变量名称 后面为变量类型
declare   @id int  
 
--循环游标
fetch next from my_cursor into @name,@id  --获取my_cursor的下一条数据,其中字段内容分别赋值给@name 和@id
while @@FETCH_STATUS=0 --假如检索到了数据继续执行  @@FETCH_STATUS是一个函数
 
begin
 
print(@name)  --这里根据业务需求定义增删改等操作
 
fetch next from my_cursor1 into @name,@id --获取下一条数据并赋值给变量
 
end--关闭释放游标
close my_cursor   --关闭游标
deallocate my_cursor   --删除游标

下面我用一张student学生表来举例说明一下

有一个student表,表中有每个学生某次考试的语文、数学、英语三课成绩,现在要在表中再加一个字段rank,统计每个人的及格科目数量

解决方法有几种,这里我用最好理解的一种方法

1、定义几个变量,存放各科成绩

2、定义rank,默认初始值为0

3、逐一判断语文数学英语成绩是否及格,这里设定的80分为及格。如果及格,rank加一

4、将rank值更新到表中

示例代码如下:


--创建一个游标
declare my_cursor1 cursor for     --my_cursor为游标的名称,随便起
select id,SLanguage,English,Maths from [student]
--打开游标
open my_cursor1                  
--变量
declare   @rank int 
declare   @Chinese int 
declare   @English int 
declare   @Math int 
declare   @id int     
--循环游标
fetch next from my_cursor1 into @id,@Chinese,@English,@Math 
while @@FETCH_STATUS=0 
begin
set @rank=0  --为变量赋初始值
if @Chinese>=80 set @rank=@rank+1
if @English>=80 set @rank=@rank+1
if @Math>=80 set @rank=@rank+1
update [student] set [rank]=@rank where id=@id
 
fetch next from my_cursor1 into @id,@Chinese,@English,@Math  --获取下一条数据并赋值给变量
 
end--关闭释放游标
close my_cursor1   --关闭游标
deallocate my_cursor1   --删除游标

获得的数据:

最后的总结

游标就是我们平常学的指针,它可以随意的移动到指定的行!!但是我们需要注意游标操作可能会引入一些性能开销,所以我们在使用游标的时候要结合性能问题去考虑

### SQL Server游标的作用和用途 #### 游标的概念 游标(Cursor)是一种用于遍历结果集的机制,允许逐行处理查询返回的结果集。它并不是简单的 `SELECT` 语句,而是一个存储在数据库管理系统 (DBMS) 上的对象,能够提供对结果集中每一行数据的操作能力[^2]。 #### 静态游标与动态游标的区别 静态游标 (`STATIC`) 是一种缓存型游标,在其创建时会将整个结果集复制到临时表中。因此,当通过静态游标访问数据时,不会反映其他事务对该数据所做的更改[^1]。 相比之下,动态游标 (`DYNAMIC`) 不会在内存中保存完整的副本,而是实时读取基础表中的数据变化。这意味着如果底层数据发生变化,则这些变更会被立即体现在游标中[^1]。 #### 创建和使用游标的语法结构 以下是标准的游标操作流程: 1. **声明游标**: 定义游标及其对应的查询逻辑。 ```sql DECLARE my_cursor CURSOR FOR SELECT name, id FROM table; ``` 2. **打开游标**: 将游标初始化以便后续迭代。 ```sql OPEN my_cursor; ``` 3. **声明变量并提取数据**: 准备接收每条记录值的局部变量,并从中取出当前行的内容。 ```sql FETCH NEXT FROM my_cursor INTO @name, @id; WHILE (@@FETCH_STATUS = 0) BEGIN PRINT(@name); FETCH NEXT FROM my_cursor INTO @name, @id; END; ``` 4. **关闭及销毁游标**: 当完成所有必要的处理后,记得清理资源以避免潜在性能问题。 ```sql CLOSE my_cursor; DEALLOCATE my_cursor; ``` #### 常见的应用场景 - **复杂业务逻辑下的单行更新**:某些情况下可能需要基于特定条件逐一修改大量独立记录;此时利用游标可实现精确控制而不必担心批量 DML 的副作用[^3]。 - **分页显示大数据量表格**:尽管现代方法更倾向于采用窗口函数等方式解决此类需求,但在早期版本或者特殊环境中仍需依赖于显式的游标技术来逐步加载页面所需部分信息。 - **异步任务调度管理**:例如定时器触发的任务队列监控程序可能会运用游标依次检查待办事项列表状态进而决定下一步动作序列。 #### 性能考量 需要注意的是,虽然游标提供了极大的灵活性,但由于它们本质上属于过程化编程风格而非集合运算方式,所以在大规模数据处理方面往往效率较低。对于大多数常规应用而言,推荐优先考虑 SET-based 方法替代 Cursor-based 解决方案除非确实存在无法规避的技术障碍才转而求助于此工具。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值