设为首页 收藏本站
查看: 744|回复: 0

[经验分享] DB2游标

[复制链接]

尚未签到

发表于 2016-11-12 10:19:11 | 显示全部楼层 |阅读模式
游标一般用来迭代结果集中的行
为了在一个过程中处理一个游标的结果,需要做以下事情:
在存储过程块的开头部分 DECLARE 游标。
打开该游标。
将游标的结果取出到之前已声明的本地变量中(隐式游标处理除外,在下面的 FOR 语句中将对此加以解释)。
关闭该游标。(注意:如果现在不关闭游标,当过程终止时将隐式地关闭游标)。
注:游标的申明如果放在中间段,要用”begin。。。end;”.段分割标志分割开;
 
游标使用的步骤如下:
1、说明游标。说明游标的时候并不执行select语句。
declare <游标名> cursor for <select语句>;
2、打开游标。打开游标实际上是执行相应的select语句,把查询结果读取到缓冲区中。这时候游标处于活动状态,指针指向查询结果集的第一条纪录。
open <游标名>;
3、推进游标指针并读取当前纪录。用fetch语句把游标指针向前推进一条纪录,同时将缓冲区中的当前纪录读取出来送到变量中。fetch语句通常用在一个循环结构体中,通过循环执行fetch语句逐条取出结果集中的行进行处理。现在好多数据库中,还允许任意方向任意步长易懂游标指针,而不仅仅是把游标指针向前推进一行了。
fetch <游标名> into <变量1>,<变量2>...
4、关闭游标。用close语句关闭游标,释放结果集占用的缓冲区及其他资源。游标关闭后,就不再和原来的查询结果集相联系。但游标可以再次打开,与新的查询结果相联系。
close <游标名>;
 
基本结构:
定义游标:
DECLARE  游标名 CURSOR FOR
Select 语句;
打开游标:
OPEN 游标名;
取值:
FETCH 游标名 INTO 变量列表
 
游标例子:
--先插入测试数据
create table test(id int,city char(20))
insert into  test values(1,'wuhan'),(2,'hangzhou'),(3,'chengdu')
 
create procedure Test(
out v_message varchar(500)
)
LANGUAGE SQL
BEGIN
DECLARE v_city char(20);
DECLARE v_count int;
 
SET v_message = '';
select count(*) into v_count from test;
BEGIN
DECLARE v_CUR CURSOR FOR SELECT city FROM test FOR READ ONLY;
OPEN v_CUR;
WHILE v_count > 0 DO
FETCH v_CUR INTO v_city;
set v_message = v_message ||v_city||' ';
set v_count = v_count -1;
end while;
END;
 
END@
运行结果为:
call Test(?)
completed successfully.
 
输出参数的值
--------------------------
参数名: V_MESSAGE
参数值: wuhan                hangzhou             chengdu             
返回状态 = 0
Statement processed successfully in 4.39 secs.
 
除了这种结构外,还有一种使用for的游标的结构,例子如下:
create procedure Test(
out v_message varchar(500)
)
LANGUAGE SQL
BEGIN
DECLARE v_city char(20);
DECLARE v_count int;
 
SET v_message = '';
 
FOR V1 AS CURSOR1 CURSOR FOR  select city as v_city from test
DO
set v_message = v_message||v_city||' ';
END FOR;  
 
END@
运行结果:
call Test(?)
completed successfully.
 
输出参数的值
--------------------------
参数名: V_MESSAGE
参数值: wuhan                hangzhou             chengdu             
 
返回状态 = 0
Statement processed successfully in 0.18 secs.
可以看到第二种游标使用起来非常简单。但是它不能使用 with hold 选项,这个with hold有什么用呢?默认情况下,,CommitRollback,游标将被关闭。所以如果游标循环体内有CommitRollback时,不能使用for形式的游标。但是第一种游标可以使用,可以在第一种游标定义时加上with hold 选项,那么在游标循环体内CommitRollback时,游标也不会关闭。
使用CommitRollback也不会关闭的游标,如下:
DECLARE v_CUR CURSOR with hold for SELECT city FROM test FOR READ ONLY;
如果要修改游标当前记录,需要定义可修改的游标,如下:
DECLARE v_CUR CURSOR for SELECT city FROM test FOR update;;
注意:for update 不能和 GROUP BY DISTINCT ORDER BY FOR READ ONLYUNION, EXCEPT UNION ALL除外)一起使用。
 
DB2存储 过程中,除了迭代结果集中的行以外,游标还可以做更多的事情。游标还可用于将结果集返回给调用程序或其他过程。
 
WITHOUT RETURN/WITH return   选项指定游标的结果表是否用于作为从一个过程中返回的结果集。
WITH RETURN TO CALLER   选项指定将来自游标的结果集返回给调用者,后者可以是另一个过程或一个客户机应用程序。这是默认选项。
WITH RETURN TO CLIENT   选项指定将来自游标的结果集返回给客户机应用程序,绕过任何中间的嵌套过程。
若要从一个过程中返回结果集,需要:
创建一个过程,创建时指定 DYNAMIC RESULT SETS 子句。
声明游标,声明时指定 WITH RETURN 子句。
打开该游标,并使之保持 open 状态。
如果关闭该游标,则结果集将不能返回给调用者应用程序。
下例 演示了一个游标的声明,该游标从一个过程中返回一个结果集:
create procedure Test(
out v_message varchar(500)
)
DYNAMIC RESULT SETS 1
LANGUAGE SQL
BEGIN
--异常处理
DECLARE SQLCODE INT;
DECLARE v_errCode INT DEFAULT 0;
DECLARE CONTINUE HANDLER FOR SQLEXCEPTION
BEGIN
SET v_errCode = SQLCODE;
END
 
--使用游标返回结果集
BEGIN
DECLARE v_cur CURSOR WITH HOLD WITH RETURN TO CALLER FOR
SELECT * FROM test;
OPEN v_cur;
END;
 
RETURN v_errCode;
END@
运行结果:
call Test(?)
completed successfully.
 
输出参数的值
--------------------------
参数名: V_MESSAGE
参数值: -
 
 
结果集 1
--------------
 
ID          CITY               
----------- --------------------
1 wuhan              
2 hangzhou           
3 chengdu            
 
3 条记录已选择。
 
返回状态 = 0

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-299237-1-1.html 上篇帖子: DB2函数 下篇帖子: db2学习
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表