db2 游标使用
游标一般用来迭代结果集中的行
为了在一个过程中处理一个游标的结果,需要做以下事情:
在存储过程块的开头部分 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有什么用呢?默认情况下,,在Commit和Rollback时,游标将被关闭。所以如果游标循环体内有Commit或Rollback时,不能使用for形式的游标。但是第一种游标可以使用,可以在第一种游标定义时加上with hold 选项,那么在游标循环体内Commit和Rollback时,游标也不会关闭。
使用Commit和Rollback也不会关闭的游标,如下:
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 ONLY及UNION, 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
游标的使用注意点:
注意 commit和rollback
使用游标时要特别注意如果没有加with hold 选项,在Commit和Rollback时,该游标将被关闭。Commit 和Rollback有很多东西要注意。特别小心
游标的两种定义方式
一种为
declare continue handler for not found
begin
set v_notfound = 1;
end;
declare cursor1 cursor with hold for select market_code from tb_market_code for update;
open cursor1;
set v_notfound=0;
fetch cursor1 into v_market_code;
while v_notfound=0 Do
--work
set v_notfound=0;
fetch cursor1 into v_market_code;
end while;
close cursor1;
这种方式使用起来比较复杂,但也比较灵活。特别是可以使用 with hold 选项。如果循环内有commit或rollback 而要保持该cursor不被关闭,只能使用这种方式。
另一种为
pcursor1: for loopcs1 as cousor1 cursor as
select market_code as market_code
from tb_market_code
for update
do
end for;
这种方式的优点是比较简单,不用(也不允许)使用 open,fetch,close。
但不能使用with hold 选项。如果在游标循环内要使用commit,rollback则不能使用这种方式。如果没有commit或rollback的要求,推荐使用这种方式(看来For这种方式有问题)。
修改游标的当前记录的方法
update tb_market_code set market_code='0' where current of cursor1;
不过要注意将cursor1定义为可修改的游标
declare cursor1 cursor for select market_code from tb_market_code
for update;
for update 不能和 GROUP BY、 DISTINCT、 ORDER BY、 FOR READ ONLY及UNION, EXCEPT, or INTERSECT 但 UNION ALL除外)一起使用。
自己平时写的实例:
create procedure db2inst1.test_overdue()
language sql
begin
declare v_count int;
declare v_id varchar(10);
declare v_shuld_repay varchar(8);
declare v_natural_mon varchar(8);
declare v_over_mon varchar(10);
declare v_day_num varchar(10);
begin
declare v_cur cursor for select id from OVERDUE_HIS for read only;
open v_cur;
set v_count=3;
--select MONTH(to_date(ACTUAL_REPAY,'YYYYMMDD'))-MONTH(to_date(shuld_repay,'YYYYMMDD'))+1 into v_count from db2inst1.overdue_his
--fetch v_cur into v_id;
while v_count>0
do
insert into OVERDUE_HIS_TAB(id,shuld_repay,natural_mon,over_mon,day_num)
select a.id,a.shuld_repay,
a.natural_mon,
0,
days(to_date(a.natural_mon,'yyyymmdd'))-days(to_date(a.shuld_repay,'yyyymmdd')) as day_num
from (select id,
shuld_repay,
to_char(TO_DATE(YEAR(to_date(shuld_repay,'YYYYMMDD')+v_count month)||'-'||MONTH(to_date(shuld_repay,'YYYYMMDD')+v_count month)||'-'||'01','YYYYMMDD')-1 day,'YYYYMMDD') as natural_mon
from OVERDUE_HIS) a;
set v_count=v_count-1;
end while;
close v_cur;
end;
end;
db2 游标使用的更多相关文章
- DB2的简单操作
转 最近在看db2,边读边写了一些,记下来,虽然写的乱七八糟.以备后用. 这些都写的很简单.我觉得也算是一些简单的操作吧,有些也是摘自别人的blog具体是引用哪的就不太记得了. 一.DB2两种注释写法 ...
- pymysql模块使用教程
一.操作数据库模板 pymysql是Python中操作mysql的模块,(使用方法几乎和MySQLdb相同,但是在Python3中,mysqldb这个库已经不能继续使用了) 下载安装方法: 方法一. ...
- mysql--pymysql 模块
pymysql模块 一.安装 cmd中使用pip或者, pycharm中控制台选择Terminal输入下面的命令,即可安装pymysql模块 pip3 install pymysql 二. 连接,执行 ...
- 【DB2】SQL0501N FETCH 语句或 CLOSE 语句中指定的游标尚未打开,或者游标标量函数引用中的游标变量尚未打开。 SQLSTATE=24501
在DB2中建立存储过程时使用了隐式游标,在调用的时候报错如下: SQL0501N FETCH 语句或 CLOSE 语句中指定的游标尚未打开,或者游标标量函数引用中的游标变量尚未打开. SQLSTA ...
- DB2 嵌入式应用中定义游标(开放平台上)
DECLARE CURSOR statement The DECLARE CURSOR statement defines a cursor. Invocation Although an inter ...
- DB2 组内分组排序,游标使用
CREATE PROCEDURE Sys_Init_tblaccountsuser_sortid () P1: BEGIN '; ; ; ; DECLARE CUR1 CURSOR WITH RETU ...
- DB2存储过程通过游标实现批量数据处理
CREATE procedure proc_change()LANGUAGE SQLBEGIN DECLARE l_id INTEGER; DECLARE l_detail_id INTEGER; D ...
- 你搞懂 ORACLE、 SQLSERVER、MYSQL与DB2的区别了吗
ORACLE. SQLSERVER.MYSQL与DB2的区别--平台性: Oracle.MYSQL与DB2可在所有主流平台上运行: SQL Server只能在Windows下运行: --安 ...
- DB2错误码信息
00 完全成功完成 表 3 01 警告 表 4 02 无数据 表 5 07 动态 SQL 错误 表 6 08 连接异常 表 7 09 触发操作异常 表 8 0A 功能部件不受支持 表 9 ...
随机推荐
- Python之路【第十八篇】Django小项目webQQ实现
WEBQQ的实现的几种方式 1.HTTP协议特点 首先这里要知道HTTP协议的特点:短链接.无状态! 在不考虑本地缓存的情况举例来说:咱们在连接博客园的时候,当tcp连接后,我会把我自己的http头发 ...
- Linux--niaoge
鸟哥的Linux私房菜 网络基础: 那 TCP/IP 是如何运作的呢?我们就拿妳常常连上的 Yahoo 入口网站来做个说明好了,整个联机的状态可以这样看: 应用程序阶段:妳打开浏览器,在浏览器上面输入 ...
- sqlplus连接oracle失败分析和解决
背景: 多台Linux服务器需要安装Oracle客户端,实现和Oracle数据库连接做业务处理. 安装完第一台后,直接将安装的目录压缩并复制到其他几台机器上,启动sqlplus连接数据库时,一直提示输 ...
- LwIP移植和使用
LwIP移植和使用 本手册基于lwip-1.4.x编写,本人没有移植过1.4.0之前的版本,更早的版本或许有差别.如果看官发现问题欢迎联系<QQ: 937431539 email: 93743 ...
- html用户注册界面
html用户注册界面 先上一张简约的界面的效果图 这里是style里面的内容 <style> input[type]{ border: 1px solid darkorange; ba ...
- 号外!GNOME 3.22 正式发布喽!!!
导读 经过半年的努力开发,别名为“卡尔斯鲁厄”的 GNOME 3.22 正式发布了!“GNOME Software 可以安装和更新 Flatpak 软件包,GNOME Builder 则可以创建它们, ...
- jquery.print.js 打印插件
<script language="javascript" src="jquery-1.4.4.min.js"></script> &l ...
- 将字符串转换成JSON对象
import net.sf.json.*; JSONObject jsStr = JSONObject.fromObject(params); //将字符串{"id":1} int ...
- Linux Swap分区设定
因为准备在linux上面安装Oracle数据库,据说swap分区小于4G容易失败,天啦噜的我连swap分区是啥都不是非常清楚, 大概有个印象吧,但先搞大了再说. 测试证明Ubuntu14默认的swap ...
- 医学CT图像特征提取算法(matlab实现)
本科毕设做的是医学CT图像特征提取方法研究,主要是肺部CT图像的特征提取.由于医学图像基本为灰度图像,因此我将特征主要分为三类:纹理特征,形态特征以及代数特征,每种特征都有对应的算法进行特征提取. 如 ...