欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

oracle学习笔记(十八) PL/SQL 游标

程序员文章站 2022-05-23 16:55:14
游标 说明 查询结果的光标,相当于java中的一个迭代器,方便遍历操作 可使用的属性 %FOUND SQL语句查询或影响了一行或多行时为 TRUE。如:mycursor%FOUND %NOTFOUND SQL语句没有影响任何行时为 TRUE。如:mycursor%NOTFOUND %ROWCOUNT ......

游标

说明

查询结果的光标,相当于java中的一个迭代器,方便遍历操作

可使用的属性

  • %found
    sql语句查询或影响了一行或多行时为 true。如:mycursor%found
  • %notfound
    sql语句没有影响任何行时为 true。如:mycursor%notfound
  • %rowcount
    sql语句查询或影响的行数。如:mycursor%rowcount
  • %isopen
    检查游标是否打开,隐式游标始终为false。如:mycursor%isopen

隐式游标

自动声明,自动打开,自动关闭,名称为sql
只能处理单行数据

declare
emp_t employee%rowtype;
begin
    select * into emp_t from employee where empno = 7859;   
    if sql%found then
    dbms_output.putline()
    end if;
end;
/

显式游标

遍历输出

cursor 游标名称[(参数列表)] is select_statement;

--使用游标遍历查询结果并输出
declare
    --给查询结果建立一个游标
    cursor cv_emp is select * from employee;
    v_emp employee%rowtype;
begin
    --使用for循环,游标会自动打开,移动和关闭
    for v_emp in cv_emp loop
        --这里只输出了员工编号和员工姓名
        dbms_output.put_line(v_emp.empno||','||v_emp.ename);
    end loop;
end;
/

游标使用

  1. 声明游标: cursor 游标名称[(参数列表)] is select_statement;
  2. 打开游标: open 游标名称[(参数列表)];
  3. 从游标中提取数据(移动游标)到变量: fetch 游标名称 into 变量列表....;
  4. 关闭游标: close 游标名称;

游标开始的时候是处于第一行之前,需要移动

--loop循环使用显示游标
--输出全部员工的信息
declare
    cursor cv_emp 
     is select empno,ename,job,sal,hiredate from employee;
    --声明一个可存储游标行的变量
    rec_emp cv_emp%rowtype;
    i number := 1;
begin
    open cv_emp;
    loop
        --会使游标向下移动,并将游标所指向的数提取到指定的变量中
        fetch cv_emp into rec_emp;
        exit when cv_emp%notfound;--找不到数据就退出循环
        dbms_output.put_line(i||'. '||rec_emp.empno||', '
                           ||rec_emp.ename||', '
                           ||rec_emp.job||','
                           ||rec_emp.sal||','
                           ||rec_emp.hiredate);
        i := i+1;
    end loop;
    close cv_emp;--关闭游标
end;
/

--使用while循环
declare
   --1. 声明游标
   cursor cv_emp 
     is select empno,ename,job,sal,hiredate from employee;
     
   rec_emp cv_emp%rowtype;
   i number := 1;
begin
    open cv_emp;--2.打开游标
    fetch cv_emp into rec_emp;--进入循环前就得移动游标
    while cv_emp%found loop
        dbms_output.put_line(i||'. '||rec_emp.empno||', '
                           ||rec_emp.ename||', '
                           ||rec_emp.job||','
                           ||rec_emp.sal||','
                           ||rec_emp.hiredate);
         fetch cv_emp into rec_emp;
         i := i+1;
    end loop;
    close cv_emp;--关闭游标
end;
/

带参数游标使用

--对所有的"销售员"(salesman)增加佣金500.
declare
    cursor cv_emp(v_job employee.job%type) is select * from employee where job = v_job;
    v_emp employee%rowtype;
begin
    for v_emp in cv_emp('salesman') loop
        --使用主键约束,更新特定行
        update employee set sal = sal +500 where empno = v_emp.empno;
    end loop;
commit; 
end;
/

游标更新数据

一般更新操作,需要一个条件,来指定需要更新的该条数据,一般使用主键来指定。
如果不想使用主键来指定,还可以使用游标当作指定条件,这就是游标更新行

--对所有的"销售员"(salesman)增加佣金500.

--普通方式更新数据
declare
    cursor cv_emp(v_job employee.job%type) is select * from employee where job = v_job;
    v_emp employee%rowtype;
begin
    for v_emp in cv_emp('salesman') loop
        --使用主键约束,更新特定行
        update employee set sal = sal +500 where empno = v_emp.empno;
    end loop;
commit; 
end;
/

--使用游标更新数据
declare
    cursor cv_emp(v_job employee.job%type) is select * from employee where job = v_job for update;
    v_emp employee%rowtype;
begin
    open cv_emp('salesman');
    loop
        fetch cv_emp into v_emp;
        --使用游标定位特定行
        update employee set sal = sal +500 where current of cv_emp;
        end if;
        exit when ev_emp%notfound;
    end loop;
    commit; 
    close cv_emp;
end;
/

ref游标

引用游标,动态游标,后期补充..