您的位置:首页 > 数据库

PLSql之游标(cursor)重点

2017-09-21 10:32 134 查看
(一)游标(cursor):在 PL/SQL 程序中,对于处理多行记录的事务经常使用游标来实现。

       
在PL/SQL块中执行SELECT、INSERT、DELETE和UPDATE语句时,ORACLE会在内存中为其分配上下文区(Context Area),即缓冲区。游标是指向该区的一个指针,或是命名一个工作区(Work Area),或是一种结构化数据类型。比较类似于java中迭代器(iterator)。


(二)游标的类型

1、隐式游标:在 PL/SQL 程序中执行DML SQL 语句时自动创建隐式游标,名字固定叫sql。
2、显式游标:显式游标用于处理返回多行的查询。
3、REF 游标:REF 游标用于处理运行时才能确定的动态 SQL 查询的结果
一、隐式游标:
       在PL/SQL中使用DML语句时自动创建隐式游标,隐式游标自动声明、打开和关闭,其名为 SQL,通过检查隐式游标的属性可以获得最近执行的 DML 语句的信息,隐式游标的属性有: %FOUND – SQL 语句影响了一行或多行时为 TRUE,%NOTFOUND – SQL
语句没有影响任何行时为TRUE,%ROWCOUNT – SQL 语句影响的行数,%ISOPEN - 游标是否打开,始终为false。

二、显示游标

1. 显式游标处理

显式游标处理需四个 PL/SQL步骤:

定义/声明游标:就是定义一个游标名,以及与其相对应的SELECT 语句。

格式:

 

    CURSOR cursor_name[(parameter[, parameter]…)] 

           [RETURN datatype]

    IS 

        select_statement;

 

游标参数只能为输入参数,其格式为: 

 

parameter_name [IN] datatype [{:= | DEFAULT} expression]

 

在指定数据类型时,不能使用长度约束。如NUMBER(4),CHAR(10) 等都是错误的。

[RETURN datatype]是可选的,表示游标返回数据的数据。如果选择,则应该严格与select_statement中的选择列表在次序和数据
4000
类型上匹配。一般是记录数据类型或带“%ROWTYPE”的数据。

 

打开游标:就是执行游标所对应的SELECT 语句,将其查询结果放入工作区,并且指针指向工作区的首部,标识游标结果集合。如果游标查询语句中带有FOR UPDATE选项,OPEN 语句还将锁定数据库表中游标结果集合对应的数据行。

格式:

 

OPEN cursor_name[([parameter =>] value[, [parameter =>] value]…)];

 

在向游标传递参数时,可以使用与函数参数相同的传值方法,即位置表示法和名称表示法。PL/SQL 程序不能用OPEN 语句重复打开一个游标。

 

提取游标数据:就是检索结果集合中的数据行,放入指定的输出变量中。 

格式:

 

FETCH cursor_name INTO {variable_list | record_variable };

 

 

执行FETCH语句时,每次返回一个数据行,然后自动将游标移动指向下一个数据行。当检索到最后一行数据时,如果再次执行FETCH语句,将操作失败,并将游标属性%NOTFOUND置为TRUE。所以每次执行完FETCH语句后,检查游标属性%NOTFOUND就可以判断FETCH语句是否执行成功并返回一个数据行,以便确定是否给对应的变量赋了值。

l 对该记录进行处理;

l 继续处理,直到活动集合中没有记录;

关闭游标:当提取和处理完游标结果集合数据后,应及时关闭游标,以释放该游标所占用的系统资源,并使该游标的工作区变成无效,不能再使用FETCH 语句取其中数据。关闭后的游标可以使用OPEN 语句重新打开。

格式:

 

CLOSE cursor_name;

 

     注:定义的游标不能有INTO 子句。

declare

   cursor c is

            select * from emp;

   v_temp c%rowtype;

begin

    open c;

    fetch c into v_temp;

    dbms_output.put_line(v_temp.ename);

    close c;

end;

------------------

declare

    cursor c is

       select * from emp;

    v_emp c%rowtype;

begin

    open c;

    loop

      fetch c into v_emp;

      exit when (c%notfound);

      dbms_output.put_line(v_emp.ename);

    end loop;

    close c;

end;

----------------------

declare

    cursor c is

       select * from emp;

    v_emp c%rowtype;

begin

    open c;

    fetch c into v_emp;

    while (c%found) loop

      dbms_output.put_line(v_emp.ename);

      fetch c into v_emp;

    end loop;

    close c;

end;

-----------------
使用游标建议使用for循环(因为for循环不用fetch,)
declare
    cursor c is
       select * from emp;
begin
   for v_emp in c loop
        dbms_output.put_line(v_emp.ename);
    end loop;
end;

--带参数的游标
declare
   cursor c (v_deptno emp.deptno%type, v_job emp.job%type)
   is
     select ename, sal from emp where deptno = v_deptno and job = v_job;
begin
   for v_temp in c(30,'CLERK') loop
      dbms_output.put_line(v_temp.ename);
   end loop;
end;

--可更新的游标
declare
  cursor c
  is
    select * from emp2 for update;
begin
   for v_temp in c loop
      if (v_temp.sal < 2000) then
         update emp2 set sal = sal * 2 where current of c;
      elsif (v_temp.sal = 5000) then
         delete from emp2 where current of c;
      end if;
    end loop;
    commit;
end;

--For 循环游标
--(1)定义游标
--(2)定义游标变量
--(3)使用for循环来使用这个游标
declare
--类型定义
cursor c_job
is
select empno,ename,job,sal
from emp
where job='MANAGER';
--定义一个游标变量v_cinfo c_emp%ROWTYPE ,该类型为游标c_emp中的一行数据类型
c_row c_job%rowtype;
begin
for c_row in c_job loop
dbms_output.put_line(c_row.empno||'-'||c_row.ename||'-'||c_row.job||'-'||c_row.sal);
end loop;
end;

--Fetch游标
--使用的时候必须要明确的打开和关闭

declare
--类型定义
cursor c_job
is
select empno,ename,job,sal
from emp
where job='MANAGER';
--定义一个游标变量
c_row c_job%rowtype;
begin
open c_job;
loop
--提取一行数据到c_row
fetch c_job into c_row;
--判读是否提取到值,没取到值就退出
--取到值c_job%notfound 是false
--取不到值c_job%notfound 是true
exit when c_job%notfound;
dbms_output.put_line(c_row.empno||'-'||c_row.ename||'-'||c_row.job||'-'||c_row.sal);
end loop;
--关闭游标
close c_job;
end;


参考文档:

http://www.cnblogs.com/huyong/archive/2011/05/04/2036377.html

http://www.cnblogs.com/sc-xx/archive/2011/12/03/2275084.html
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: