cursor cursor_name(parameter_name datatype) is select_statement;
定义参数游标时,游标参数只能指定数据类型,而不能指定长度。
示例如下:
Oracle代码
- declare
- cursor temp_cursor(no number) is select name from cip_temps where id=no;
- v_name cip_temps.name%type;
- begin
- open temp_cursor(1
); - loop
- fetch temp_cursor into v_name;
- exit when temp_cursor%notfound;
- dbms_output.put_line(v_name);
- end loop;
- close temp_cursor;
- end;
通过使用显示游标,不仅可以一行一行的处理select语句结果,而且也可以更新或删除当前游标的数据,注意,如果要通过游标更新或删除数据,在定义游标时一定要带有for update子句,语法如下:
cursor cursor_name(parameter_name datatype) is select_statement for updae [of column_reference][nowait];
update table_name set column=.. where current of cursor_name;
delete from table_name where current of cursor_name;
- declare
Oracle代码
- declare
- cursor temp_cursor is select name,address,id from cip_temps for update;
- v_name cip_temps.name%type;
- v_address cip_temps.ADDRESS%type;
- v_id cip_temps.id%type;
- begin
- open temp_cursor;
- loop
- fetch temp_cursor into v_name,v_address,v_id;
- exit when temp_cursor%NOTFOUND;
- if(v_id>2
) then - delete from cip_temps where current of temp_cursor;
- end if;
- end loop;
- close temp_cursor;
- end;
如果使用子查询涉及到多张表,那么默认情况下会在所有表上加行共享锁,为了只在特定表上加行共享锁,需要在for update子句后带有of子句,of后面跟字段名,如果跟表名或游标名称,则会报错:标示符无效。示例如下:
Oracle代码
- declare
- cursor gData is select name,address,cip_temps.id from cip_temps,cip_t
- where cip_temps.id=cip_t.id for update of address;
- rs gData%rowtype;
- begin
- open gData;
- loop
- fetch gData into rs;
- exit when gData%notfound;
- if rs.id=1
then - delete from cip_temps where current of gData;
- else
- update cip_temps set name='king'
where current of gData; - end if;
-
- end loop;
- close gData;
- end;
使用for update语句对被作用于行加锁,如果其他会话已经在被作用于行上加锁,那么默认情况下当前会话要一直等待对方释放锁,通过在for update子句中指定 nowait语句,可以避免等待锁,当指定了nowait子句之后,如果其他会话已经在被作用行加锁,那么当前会话会显示错误提示信息,并退出PL/SQL,示例如下:
Oracle代码
- declare
- cursor gData is select name,address,cip_temps.id from cip_temps,cip_t
- where cip_temps.id=cip_t.id for update nowait;
- rs gData%rowtype;
- begin
- open gData;
- loop
- fetch gData into rs;
- exit when gData%notfound;
- if rs.id=1
then - delete from cip_temps where current of gData;
- else
- update cip_temps set name='king'
where current of gData; - end if;
-
- end loop;
- close gData;
- end;
三、游标for循环
使用游标for循环是循环游标最简单的方法,oracle会隐含打开游标、循环提取数据、关闭游标,语法如下:
for record_name in cursor_name loop
..........
end loop;
如上所示:cursor_name是已经定义的游标名称,record_name是oracle隐含定义的记录变量。
1、使用游标for循环
当使用游标开发程序时,建议使用for循环,从而简化代码程序,示例如下:
Oracle代码
- declare
- cursor temp_cursor is select name,age,address,id from cip_temps;
- begin
- for emp_record in temp_cursor loop
- dbms_output.put_line(temp_cursor%rowcount||'第一行数据:'
||emp_record.name||':' || emp_record.age||':' || emp_record.address||':' || emp_record.id); - end loop;
- end;
2、在游标for循环时直接使用子查询
Sql代码
- declare
- begin
- for
emp_record in (select * from cip_temps) loop - dbms_output.put_line('第一行数据:'
||emp_record.name ||':' || emp_record.age||':' || emp_record.address||':' || emp_record.id); - end
loop; - end
;
本文转自My_King1 51CTO博客,原文链接:http://blog.51cto.com/apprentice/1360599,如需转载请自行联系原作者