SQL Server之游标的基础知识

本文涉及的产品
云数据库 RDS SQL Server,基础系列 2核4GB
RDS SQL Server Serverless,2-4RCU 50GB 3个月
推荐场景:
简介: 什么是游标: 游标是可以在结果集中上下游动的指针。 游标的作用: --允许定位到结果集中的特定行。 --从结果集的当前位置检索一行或多行数据。 --支持对结果集中当前位置的行进行修改。 注意:游标虽然很好用,但是如果滥用游标的话,会对程序的性能造成很大影响,使用的时候一定要谨慎啊! 游标的种类: MS SQL SERVER 支持三种类型的游标:Transact_SQL 游标,API 服务器游标和客户游标。

什么是游标: 游标是可以在结果集中上下游动的指针。

游标的作用: --允许定位到结果集中的特定行。 --从结果集的当前位置检索一行或多行数据。 --支持对结果集中当前位置的行进行修改。 注意:游标虽然很好用,但是如果滥用游标的话,会对程序的性能造成很大影响,使用的时候一定要谨慎啊!

游标的种类: MS SQL SERVER 支持三种类型的游标:Transact_SQL 游标,API 服务器游标和客户游标。

(1) Transact_SQL 游标 Transact_SQL 游标是由declare cursor语法定义、主要用在Transact_SQL 脚本、存储过程和触发器中。 Transact_SQL 游标主要用在服务器上,由从客户端发送给服务器的Transact_SQL 语句或是批处理、存储过程、触发器中的Transact_SQL 进行管理。 Transact_SQL 游标不支持提取数据块或多行数据。

(2) API 游标 API 游标支持在OLE DB, ODBC 以及DB_library 中使用游标函数,主要用在服务器上。 每一次客户端应用程序调用API 游标函数,MS SQL SEVER 的OLE DB 提供者、ODBC驱动器或DB_library 的动态链接库(DLL) 都会将这些客户请求传送给服务器以对API游标进行处理。

(3) 客户游标 客户游标主要是当在客户机上缓存结果集时才使用。在客户游标中,有一个缺省的结果集被用来在客户机上缓存整个结果集。 客户游标仅支持静态游标而非动态游标。 由于服务器游标并不支持所有的Transact-SQL 语句或批处理,所以客户游标常常仅被用作服务器游标的辅助。 因为在一般情况下,服务器游标能支持绝大多数的游标操作。 由于API 游标和Transact-SQL 游标使用在服务器端,所以被称为服务器游标,也被称为后台游标,而客户端游标被称为前台游标。

游标的使用方法: 使用游标有四种基本的步骤:声明游标、打开游标、提取数据、关闭游标。

声明一个游标 国际标准语句(ISO Syntax): declare cursor_name { [insensitive ]| [scroll] } cursor for sql_sentence [ for{ read only | update[ of column_name [ ,...n ] ] } ]

Transact-SQL 扩展语句(Transact-SQL Extended Syntax): declare cursor_name cursor [ local| global] [ forward_only | scroll] [ static| keyset| dynamic | fast_forward ] [ read_only | scroll_locks | optimistic ] [ type_warning ] for select_statement [ for update[ of column_name [ ,...n ] ] ]

insensitive 定义一个游标,以创建将由该游标使用的数据的临时复本。对游标的所有请求都从 tempdb 中的这一临时表中得到应答; 因此,在对该游标进行提取操作时返回的数据中不反映对基表所做的修改,并且该游标不允许修改。 使用国际语法时,如果省略 insensitive ,则已提交的(任何用户)对基础表的删除和更新则会反映在后面的提取操作中。

scroll 指定所有的提取选项(first、last、prior、next、relative、absolute)均可用。 如果未在 ISO declare cursor 中指定 scroll,则 next 是唯一支持的提取选项。 如果也指定了 fast_forward,则不能指定 scroll。

read only 禁止通过该游标进行更新。在 update或 delete语句的 where current of 子句中不能引用该游标。 该选项优于要更新的游标的默认功能。 update[ of column_name [ ,...n ] ] 定义游标中可更新的列。如果指定了 of column_name [,...n],则只允许修改所列出的列。 如果指定了 update,但未指定列的列表,则可以更新所有列。 local 指定对于在其中创建的批处理、存储过程或触发器来说,该游标的作用域是局部的。 该游标名称仅在这个作用域内有效。在批处理、存储过程、触发器或存储过程 output参数中,该游标可由局部游标变量引用。 output参数用于将局部游标传递回调用批处理、存储过程或触发器,它们可在存储过程终止后给游标变量分配参数使其引用游标。 除非 output参数将游标传递回来,否则游标将在批处理、存储过程或触发器终止时隐式释放。 如果 output参数将游标传递回来,则游标在最后引用它的变量释放或离开作用域时释放。

global 指定该游标的作用域对来说连接是全局的。 在由连接执行的任何存储过程或批处理中,都可以引用该游标名称。该游标仅在断开连接时隐式释放。

forward_only 指定游标只能从第一行滚动到最后一行。fetch next 是唯一支持的提取选项。 如果在指定forward_only时不指定 static、keyset和dynamic 关键字,则游标作为 dynamic 游标进行操作。 如果forward_only和 scroll均未指定,则除非指定 static、keyset或dynamic 关键字,否则默认为forward_only。 static、keyset和dynamic 游标默认为scroll。 与 ODBC 和 ADO 这类数据库 API 不同,static、keyset或dynamic Transact-SQL 游标支持 forward_only。

static 定义一个游标,以创建将由该游标使用的数据的临时复本。 对游标的所有请求都从 tempdb 中的这一临时表中得到应答; 因此,在对该游标进行提取操作时返回的数据中不反映对基表所做的修改,并且该游标不允许修改。

keyset 指定当游标打开时,游标中行的成员身份和顺序已经固定。对行进行唯一标识的键集内置在 tempdb 内一个称为 keyset 的表中。

dynamic 定义一个游标,以反映在滚动游标时对结果集内的各行所做的所有数据更改。 行的数据值、顺序和成员身份在每次提取时都会更改。动态游标不支持 absolute 提取选项。

fast_forward 指定启用了性能优化的 forward_only、read_only 游标。如果指定了scroll或for_update,则不能也指定fast_forward。

注意:在 SQL Server 2000 中,FAST_FORWARD 和 FORWARD_ONLY 游标选项是互相排斥的。 如果指定了二者,则会引发错误。在 SQL Server 2005 及更高版本中,这两个关键字可以用在同一个 DECLARE CURSOR 语句中。

scroll_locks 指定通过游标进行的定位更新或删除一定会成功。将行读入游标时 SQL Server 将锁定这些行,以确保随后可对它们进行修改。 如果还指定了fast_forward或static,则不能指定scroll_locks。

optimistic 指定如果行自读入游标以来已得到更新,则通过游标进行的定位更新或定位删除不成功。 当将行读入游标时,SQL Server 不锁定行。 它改用 timestamp 列值的比较结果来确定行读入游标后是否发生了修改,如果表不含 timestamp 列,它改用校验和值进行确定。 如果已修改该行,则尝试进行的定位更新或删除将失败。如果还指定了fast_forward,则不能指定optimistic。

type_warning 指定将游标从所请求的类型隐式转换为另一种类型时向客户端发送警告消息。

打开游标: open cursor_name 由于打开游标是对数据库进行一些SQL SELECT的操作,它将耗费一段时间,主要取决于您使用的系统性能和这条语句的复杂程度。

提取游标:

fetch ** from cursor_name into 变量

当用OPEN语句打开了游标并在数据库中执行了查询后,您不能立即利用在查询结果集中的数据。

您必须用FETCH语句来取得数据。 一条FETCH语句一次可以将一条记录放入程序员指定的变量中。 

--fetch first:提取游标的第一行。

--fetch next:提取上次提取的行的下一行。

--fetch prior:提取上次提取的行的前一行。

--fetch last:提取游标中的最后一行。

--fetch absolute n:

-- 如果n 为正整数,则提取 游标中的第n行

-- 如果n为负整数,则提取游标最后一行之前的第n行

-- 如果n 为0,则不提取任何行

--fetch relative n :

-- 如果n为正,则提取上次提取的行之后的第n行。

-- 如果n为负,则提取上提取的行之前的第n行。

-- 如果n为0,则再次提取同一行

@@fetch_status,返回针对连接当前打开的任何游标发出的上一条游标 fetch语句的状态。  0  fetch语句成功 -1  fetch语句失败或行不在结果集中 -2  提取的行不存在

关闭游标 close cursor_name 删除游标资源,释放内存 deallocate  cursor_name

一个简单的游标示例:

 

create proc pro_cursor
as
begin
--声明一个全局游标
declare mycursor cursor for 
select sid from score 
--打开游标
open mycursor
--声明一个变量
declare @sid int
--循环移动
fetch next from mycursor into @sid
while(@@fetch_status=0)
  begin
    update score set score=score+10
where sid=@sid
    fetch next from mycursor into @sid
  end
close mycursor
deallocate mycursor
end

 

 

原文:http://www.cnblogs.com/jiajiayuan/archive/2011/07/14/2106341.html

 

相关实践学习
使用SQL语句管理索引
本次实验主要介绍如何在RDS-SQLServer数据库中,使用SQL语句管理索引。
SQL Server on Linux入门教程
SQL Server数据库一直只提供Windows下的版本。2016年微软宣布推出可运行在Linux系统下的SQL Server数据库,该版本目前还是早期预览版本。本课程主要介绍SQLServer On Linux的基本知识。 相关的阿里云产品:云数据库RDS SQL Server版 RDS SQL Server不仅拥有高可用架构和任意时间点的数据恢复功能,强力支撑各种企业应用,同时也包含了微软的License费用,减少额外支出。 了解产品详情: https://www.aliyun.com/product/rds/sqlserver
相关文章
|
2月前
|
关系型数据库 MySQL 网络安全
5-10Can't connect to MySQL server on 'sh-cynosl-grp-fcs50xoa.sql.tencentcdb.com' (110)")
5-10Can't connect to MySQL server on 'sh-cynosl-grp-fcs50xoa.sql.tencentcdb.com' (110)")
|
4月前
|
SQL 存储 监控
SQL Server的并行实施如何优化?
【7月更文挑战第23天】SQL Server的并行实施如何优化?
98 13
|
4月前
|
SQL
解锁 SQL Server 2022的时间序列数据功能
【7月更文挑战第14天】要解锁SQL Server 2022的时间序列数据功能,可使用`generate_series`函数生成整数序列,例如:`SELECT value FROM generate_series(1, 10)。此外,`date_bucket`函数能按指定间隔(如周)对日期时间值分组,这些工具结合窗口函数和其他时间日期函数,能高效处理和分析时间序列数据。更多信息请参考官方文档和技术资料。
|
4月前
|
SQL 存储 网络安全
关系数据库SQLserver 安装 SQL Server
【7月更文挑战第26天】
59 6
|
4月前
|
存储 SQL C++
对比 SQL Server中的VARCHAR(max) 与VARCHAR(n) 数据类型
【7月更文挑战7天】SQL Server 中的 VARCHAR(max) vs VARCHAR(n): - VARCHAR(n) 存储最多 n 个字符(1-8000),适合短文本。 - VARCHAR(max) 可存储约 21 亿个字符,适合大量文本。 - VARCHAR(n) 在处理小数据时性能更好,空间固定。 - VARCHAR(max) 对于大文本更合适,但可能影响性能。 - 选择取决于数据长度预期和业务需求。
306 1
|
3月前
|
SQL 安全 Java
驱动程序无法通过使用安全套接字层(SSL)加密与 SQL Server 建立安全连接。错误:“The server selected protocol version TLS10 is not accepted by client
驱动程序无法通过使用安全套接字层(SSL)加密与 SQL Server 建立安全连接。错误:“The server selected protocol version TLS10 is not accepted by client
432 0
|
4月前
|
SQL Oracle 关系型数据库
MySQL、SQL Server和Oracle数据库安装部署教程
数据库的安装部署教程因不同的数据库管理系统(DBMS)而异,以下将以MySQL、SQL Server和Oracle为例,分别概述其安装部署的基本步骤。请注意,由于软件版本和操作系统的不同,具体步骤可能会有所变化。
276 3
|
4月前
|
SQL 存储 安全
数据库数据恢复—SQL Server数据库出现逻辑错误的数据恢复案例
SQL Server数据库数据恢复环境: 某品牌服务器存储中有两组raid5磁盘阵列。操作系统层面跑着SQL Server数据库,SQL Server数据库存放在D盘分区中。 SQL Server数据库故障: 存放SQL Server数据库的D盘分区容量不足,管理员在E盘中生成了一个.ndf的文件并且将数据库路径指向E盘继续使用。数据库继续运行一段时间后出现故障并报错,连接失效,SqlServer数据库无法附加查询。管理员多次尝试恢复数据库数据但是没有成功。
|
4月前
|
SQL 存储 关系型数据库
关系型数据库SQL Server学习
【7月更文挑战第4天】
72 2
|
5月前
|
SQL 存储 测试技术