SQL游标使用及实例分析
declare my_cursor cursor scroll dynamic
/*scroll表示可随意移动游标指针(否则只能向前),dynamic表示可以读写游标(否则游标只读)*/
for
select * from t_msg
open my_cursor
declare @name sysname
fetch next from my_cursor into @name
while(@@fetch_status=0)
begin
print 'username: ' + @name
--fetch next from my_cursor
fetch next from my_cursor into @name
end
--fetch first from my_cursor into @name
print @name
/* update 个人资料 set 姓名='zzg' where current of my_cursor */
/* delete from 个人资料 where current of my_cursor */
close my_cursor
deallocate my_cursor
使用游标(cursor)的一个主要的原因就是把集合操作转换成单个记录处理方式。用sql语言从中检索数据后,结果放在内存的一块区域中,且结果往往是一个含有多个记录的集合。游标机制允许用户在sql server内逐行地访问这些记录,按照用户自己的意愿来显示和处理这些记录。
2. 如何使用游标:
一般地,使用游标都遵循下列的常规步骤:
(1) 声明游标。把游标与t-sql语句的结果集联系起来。
(2) 打开游标。
(3) 使用游标操作数据。
(4) 关闭游标。
2.1. 声明游标
declare cursor语句sql-92标准语法格式:
declare 游标名 [ insensitive ] [ scroll ] cursor
for sql-statement
eg:
declare mycrsrvar cursor
for select * from tbmydata
2.2 打开游标
open mycrsrvar
当游标被打开时,行指针将指向该游标集第1行之前,如果要读取游标集中的第1行数据,必须移动行指针使其指向第1行。就本例而言,可以使用下列操作读取第1行数据:
fetch first from e1cursor
或 fetch next from e1cursor
2.3 使用游标操作数据
下面的示例用@@fetch_status控制在一个while循环中的游标活动
/* 使用游标读取数据的操作如下。*/
declare e1cursor cursor /* 声明游标,默认为forward_only游标 */
for select * from c_example
open e1cursor /* 打开游标 */
fetch next from e1cursor /* 读取第1行数据*/
while @@fetch_status = 0 /* 用while循环控制游标活动 */
begin
fetch next from e1cursor /* 在循环体内将读取其余行数据 */
end
close e1cursor /* 关闭游标 */
deallocate e1cursor /* 删除游标 */
2.4 关闭游标
使用close语句关闭游标
close { { [ global ] 游标名 } | 游标变量名 }
使用deallocate语句删除游标,其语法格式如下:
deallocate { { [ global ] 游标名 } | @游标变量名
3. fetch操作的简明语法如下:
fetch
[ next | prior | first | last]
from
{ 游标名 | @游标变量名 } [ into @变量名 [,…] ]
参数说明:
next 取下一行的数据,并把下一行作为当前行(递增)。由于打开游标后,行指针是指向该游标第1行之前,所以第一次执行fetch next操作将取得游标集中的第1行数据。next为默认的游标提取选项。
into @变量名[,…] 把提取操作的列数据放到局部变量中。列表中的各个变量从左到右与游标结果集中的相应列相关联。各变量的数据类型必须与相应的结果列的数据类型匹配或是结果列数据类型所支持的隐性转换。变量的数目必须与游标选择列表中的列的数目一致。
--------------------------------------------------------------------------------------------------------------------------------
每执行一个fetch操作之后,通常都要查看一下全局变量@@fetch_status中的状态值,以此判断fetch操作是否成功。该变量有三种状态值:
? 0 表示成功执行fetch语句。
? -1 表示fetch语句失败,例如移动行指针使其超出了结果集。
? -2 表示被提取的行不存在。
由于@@fetch_statu是全局变量,在一个连接上的所有游标都可能影响该变量的值。因此,在执行一条fetch语句后,必须在对另一游标执行另一fetch 语句之前测试该变量的值才能作出正确的判断。
更新数据;
declare my_youbiao cursor
for select * from t_msg
for update
open my_youbiao
fetch next from my_youbiao
while @@fetch_status=0
begin
--update t_msg set msg='1234567890' where current of my_youbiao
update my_youbiao set msg='123' where current of my_youbiao
fetch next from my_youbiao
print 'asdfasd11'
end
close my_youbiao
deallocate my_youbiao
print 'asdfasd'
测试通过:
--select * from master..sysprocesses
use test
declare my_cursor cursor scroll dynamic --scroll表示可以向前或向后移动 dynamic:表示可写也可读,
for
select f3 from temp --定义my_cursor 游标
open my_cursor --打开游标
declare @name nvarchar(128) --定义一个变量
fetch next from my_cursor into @name --游标停在第一条记录前面,第一次执行,测试有没有记录存在
while(@@fetch_status=0) --取数据,直到-2即没有记录
begin
print '姓名: ' + @name
--fetch next from my_cursor
fetch next from my_cursor into @name
end
--fetch first from my_cursor into @name
print @name
--update temp set f9='zzg' where current of my_cursor
/* delete from 个人资料 where current of my_cursor */
close my_cursor
deallocate my_cursor