oracle存储过程的游标使用,oracle存储过程和游标的使用

游标:sql

用来查询数据库,获取记录集合(结果集)的指针,咱们所说的游标一般是指显式游标,所以从如今起没有特别指明的状况,咱们所说的游标都是指显式游标。要在程序中使用游标,必须首先声明游标分类:数据库

静态游标:

分为显式游标和隐式游标。安全

REF游标:

是一种引用类型,相似于指针。cors

显式游标:函数

CURSOR 游标名 ( 参数 ) [返回值类型] IS

Select 语句oop

生命周期:测试

在大多数时候咱们在设计程序的时候都遵循下面的步骤:

一、打开游标 open cs1;

二、开始循环 while cs1%found loop | for column_name in .. LOOP

三、从游标中取值 fetch .. into.. |

四、检查那一行被返回

五、处理

六、关闭循环 end loop;

七、关闭游标 if cs1&isopen then close cs1;fetch

选项:参数和返回类型设计

set serveroutput on

declare

cursor emp_cur ( p_deptid in number) is

select * from employees where department_id = p_deptid;指针

l_emp employees%rowtype;

begin

dbms_output.put_line('Getting employees from department 30');

open emp_cur(30);

loop

fetch emp_cur into l_emp;

exit when emp_cur%notfound;

dbms_output.put_line('Employee id '|| l_emp.employee_id || ' is ');

dbms_output.put_line(l_emp.first_name || ' ' || l_emp.last_name);

end loop;

close emp_cur;

dbms_output.put_line('Getting employees from department 90');

open emp_cur(90);

loop

fetch emp_cur into l_emp;

exit when emp_cur%notfound;

dbms_output.put_line('Employee id '|| l_emp.employee_id || ' is ');

dbms_output.put_line(l_emp.first_name || ' ' || l_emp.last_name);

end loop;

close emp_cur;

end;

/

隐式游标:

不用明确创建游标变量,分两种:

1.在PL/SQL中使用DML语言,使用ORACLE提供的名为SQL的隐示游标

2.CURSOR FOR LOOP,用于for loop 语句

1举例:

declare

begin

update departments set department_name=department_name;

--where 1=2;

dbms_output.put_line('update '|| sql%rowcount ||' records');

end;

/

2举例:

declare

begin

for my_dept_rec in ( select department_name, department_id from departments)

loop

dbms_output.put_line(my_dept_rec.department_id || ' : ' || my_dept_rec.department_name);

end loop;

end;

/

3举例:

单独select

declare

l_empno emp.EMPLOYEE_ID%type;

-- l_ename emp.ename%type;

begin

select EMPLOYEE_ID

into l_empno

from emp;

--where rownum =1;

dbms_output.put_line(l_empno);

end;

/

使用INTO获取值,只能返回一行。

游标属性:

%FOUND:变量最后从游标中获取记录的时候,在结果集中找到了记录。

%NOTFOUND:变量最后从游标中获取记录的时候,在结果集中没有找到记录。

%ROWCOUNT:当前时刻已经从游标中获取的记录数量。

%ISOPEN:是否打开。

Declare

Cursor emps is

Select * from employees where rownum<6 order by 1;

Emp employees%rowtype;

Row number :=1;

Begin

Open emps;

Fetch emps into emp;

Loop

If emps%found then

Dbms_output.put_line('Looping over record '||row|| ' of ' || emps%rowcount);

Fetch emps into emp;

Row := row + 1;

Elsif emps%notfound then

Exit;  ---exit loop, not IF

End if;

End loop;

If emps%isopen then

Close emps;

End if;

End;

/

显式和隐式游标的区别:

尽可能使用隐式游标,避免编写附加的游标控制代码(声明,打开,获取,关闭),也不须要声明变量来保存从游标中获取的数据。

REF CURSOR游标:

动态游标,在运行的时候才能肯定游标使用的查询。分类:

强类型(限制)REF CURSOR,规定返回类型

弱类型(非限制)REF CURSOR,不规定返回类型,能够获取任何结果集。

TYPE ref_cursor_name IS REF CURSOR [RETURN return_type]

Declare

Type refcur_t is ref cursor;

Type emp_refcur_t is ref cursor return employee%rowtype;

Begin

Null;

End;

/

强类型举例:

declare

--声明记录类型

type emp_job_rec is record(

employee_id number,

employee_name varchar2(50),

job_title varchar2(30)

);

--声明REF CURSOR,返回值为该记录类型

type emp_job_refcur_type is ref cursor

return emp_job_rec;

--定义REF CURSOR游标的变量

emp_refcur emp_job_refcur_type;

emp_job emp_job_rec;

begin

open emp_refcur for

select e.employee_id,

e.first_name || ' ' ||e.last_name "employee_name",

j.job_title

from employees e, jobs j

where e.job_id = j.job_id and rownum < 11 order by 1;

fetch emp_refcur into emp_job;

while emp_refcur%found loop

dbms_output.put_line(emp_job.employee_name || '''s job is ');

dbms_output.put_line(emp_job.job_title);

fetch emp_refcur into emp_job;

end loop;

end;

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

Oracle数据库游标使用大全

SQL是用于访问ORACLE数据库的语言,PL/SQL扩展和增强了SQL的功能,它同时引入了更

强的程序逻辑。 PL/SQL支持DML命令和SQL的事务控制语句。DDL在PL/SQL中不被支持,这就

意味做在PL/SQL程序块中不能建立表或其余任何对象。较好的PL/SQL程序设计是在PL/SQL块中

使用象DBMS_SQL这样的内建包或执行EXECUTE IMMEDIATE命令创建动态SQL来执行DDL命令,

下面咱们将讨论各类用于访问ORACLE数据库的DDL和TCL语句。

查询

SELECT语句用于从数据库中查询数据,当在PL/SQL中使用SELECT语句时,要与INTO子

句一块儿使用,查询的返回值被赋予INTO子句中的变量,变量的声明是在DELCARE中。SELECT

SELECT [DISTICT|ALL]{*|column[,column,...]}

INTO (variable[,variable,...] |record)

FROM {table|(sub-query)}[alias]

WHERE............

PL/SQL中SELECT语句只返回一行数据。若是超过一行数据,那么就要使用显式游标(

对游标的讨论咱们将在后面进行),INTO子句中要有与SELECT子句中相同列数量的变量。INTO

%TYPE属性

在PL/SQL中能够将变量和常量声明为内建或用户定义的数据类型,以引用一个列名,

同时继承他的数据类型和大小。这种动态赋值方法是很是有用的,好比变量引用的列的数据类

型和大小改变了,若是使用了%TYPE,那么用户就没必要修改代码,不然就必须修改代码。

例:

v_empno SCOTT.EMP.EMPNO%TYPE;

v_salary EMP.SALARY%TYPE;

不但列名可使用%TYPE,并且变量、游标、记录,或声明的常量均可以使用%TYPE。

DELCARE

V_A NUMBER(5):=10;

V_B V_A%TYPE:=15;

V_C V_A%TYPE;

BEGIN

DBMS_OUTPUT.PUT_LINE

('V_A='||V_A||'V_B='||V_B||'V_C='||V_C);

END

SQL>/

V_A=10 V_B=15 V_C=

PL/SQL procedure successfully completed.

其余DML语句

其它操做数据的DML语句是:INSERT、UPDATE、DELETE和LOCK TABLE,这些语句在

PL/SQL中的语法与在SQL中的语法相同。咱们在前面已经讨论过DML语句的使用这里就再也不重

复了。在DML语句中可使用任何在DECLARE部分声明的变量,若是是嵌套块,那么要注意变

例:

CREATE OR REPLACE PROCEDURE FIRE_EMPLOYEE (pempno in number)

AS

v_ename EMP.ENAME%TYPE;

BEGIN

SELECT ename INTO v_ename

FROM emp

WHERE empno=p_empno;

INSERT INTO FORMER_EMP(EMPNO,ENAME)

VALUES (p_empno,v_ename);

DELETE FROM emp

WHERE empno=p_empno;

UPDATE former_emp

SET date_deleted=SYSDATE

WHERE empno=p_empno;

EXCEPTION

WHEN NO_DATA_FOUND THEN

DBMS_OUTPUT.PUT_LINE('Employee Number Not Found!');

END

DML语句的结果

当执行一条DML语句后,DML语句的结果保存在四个游标属性中,这些属性用于控制程

序流程或者了解程序的状态。当运行DML语句时,PL/SQL打开一个内建游标并处理结果,游标

是维护查询结果的内存中的一个区域,游标在运行DML语句时打开,完成后关闭。隐式游标只

使用SQL%FOUND,SQL%NOTFOUND,SQL%ROWCOUNT三个属性.SQL%FOUND,SQL%NOTFOUND是布尔值,

SQL%FOUND和SQL%NOTFOUND

在执行任何DML语句前SQL%FOUND和SQL%NOTFOUND的值都是NULL,在执行DML语句后,

. TRUE :INSERT

. TRUE :DELETE和UPDATE,至少有一行被DELETE或UPDATE.

. TRUE :SELECT INTO至少返回一行

当SQL%FOUND为TRUE时,SQL%NOTFOUND为FALSE。

SQL%ROWCOUNT

在执行任何DML语句以前,SQL%ROWCOUNT的值都是NULL,对于SELECT INTO语句,若是

执行成功,SQL%ROWCOUNT的值为1,若是没有成功,SQL%ROWCOUNT的值为0,同时产生一个异常

SQL%ISOPEN

SQL%ISOPEN是一个布尔值,若是游标打开,则为TRUE, 若是游标关闭,则为FALSE.对

于隐式游标而言SQL%ISOPEN老是FALSE,这是由于隐式游标在DML语句执行时打开,结束时就

事务控制语句

事务是一个工做的逻辑单元能够包括一个或多个DML语句,事物控制帮助用户保证数

据的一致性。若是事务控制逻辑单元中的任何一个DML语句失败,那么整个事务都将回滚,在

PL/SQL中用户能够明确地使用COMMIT、ROLLBACK、SAVEPOINT以及SET TRANSACTION语句。

COMMIT语句终止事务,永久保存数据库的变化,同时释放全部LOCK,ROLLBACK终止现

行事务释放全部LOCK,但不保存数据库的任何变化,SAVEPOINT用于设置中间点,当事务调用过

多的数据库操做时,中间点是很是有用的,SET TRANSACTION用于设置事务属性,好比

显式游标

当查询返回结果超过一行时,就须要一个显式游标,此时用户不能使用select into

语句。PL/SQL管理隐式游标,当查询开始时隐式游标打开,查询结束时隐式游标自动关闭。显

式游标在PL/SQL块的声明部分声明,在执行部分或异常处理部分打开,取数据,关闭。下表显

表1 隐式游标和显式游标

[[The No.1 Picture.]]

使用游标

这里要作一个声明,咱们所说的游标一般是指显式游标,所以从如今起没有特别指明

的状况,咱们所说的游标都是指显式游标。要在程序中使用游标,必须首先声明游标。

声明游标

语法:

CURSOR cursor_name IS select_statement;

在PL/SQL中游标名是一个未声明变量,不能给游标名赋值或用于表达式中。

例:

DELCARE

CURSOR C_EMP IS SELECT empno,ename,salary

FROM emp   WHERE salary>2000

ORDER BY ename;

........

BEGIN

在游标定义中SELECT语句中不必定非要表能够是视图,也能够从多个表或视图中选择

打开游标

使用游标中的值以前应该首先打开游标,打开游标初始化查询处理。打开游标的语法

OPEN cursor_name

cursor_name是在声明部分定义的游标名。

例:

OPEN C_EMP;

关闭游标

语法:

CLOSE cursor_name

例:

CLOSE C_EMP;

从游标提取数据

从游标获得一行数据使用FETCH命令。每一次提取数据后,游标都指向结果集的下一

FETCH cursor_name INTO variable[,variable,...]

对于SELECT定义的游标的每一列,FETCH变量列表都应该有一个变量与之相对应,变

例:

SET SERVERIUTPUT ON

DECLARE

v_ename EMP.ENAME%TYPE;

v_salary EMP.SALARY%TYPE;

CURSOR c_emp IS SELECT ename,salary FROM emp;

BEGIN

OPEN c_emp;

FETCH c_emp INTO v_ename,v_salary;

DBMS_OUTPUT.PUT_LINE('Salary of Employee'|| v_ename

||'is'|| v_salary);

FETCH c_emp INTO v_ename,v_salary;

DBMS_OUTPUT.PUT_LINE('Salary of Employee'|| v_ename

||'is'|| v_salary);

FETCH c_emp INTO v_ename,v_salary;

DBMS_OUTPUT.PUT_LINE('Salary of Employee'|| v_ename

||'is'|| v_salary);

CLOSE c_emp;

END

这段代码无疑是很是麻烦的,若是有多行返回结果,可使用循环并用游标属性为结

束循环的条件,以这种方式提取数据,程序的可读性和简洁性都大为提升,下面咱们使用循环

SET SERVERIUTPUT ON

DECLARE

v_ename EMP.ENAME%TYPE;

v_salary EMP.SALARY%TYPE;

CURSOR c_emp IS SELECT ename,salary FROM emp;

BEGIN

OPEN c_emp;

LOOP

FETCH c_emp INTO v_ename,v_salary;

EXIT WHEN c_emp%NOTFOUND;

DBMS_OUTPUT.PUT_LINE('Salary of Employee'|| v_ename

||'is'|| v_salary);

END

记录变量

定义一个记录变量使用TYPE命令和%ROWTYPE,关于%ROWsTYPE的更多信息请参阅相关

记录变量用于从游标中提取数据行,当游标选择不少列的时候,那么使用记录比为每

当在表上使用%ROWTYPE并将从游标中取出的值放入记录中时,若是要选择表中全部列

,那么在SELECT子句中使用*比将全部列名列出来要安全得多。

例:

SET SERVERIUTPUT ON

DECLARE

R_emp EMP%ROWTYPE;

CURSOR c_emp IS SELECT * FROM emp;

BEGIN

OPEN c_emp;

LOOP

FETCH c_emp INTO r_emp;

EXIT WHEN c_emp%NOTFOUND;

DBMS_OUT.PUT.PUT_LINE('Salary of Employee'||r_emp.ename||'is'|| r_emp.salary);

END LOOP;

CLOSE c_emp;

END;

%ROWTYPE也能够用游标名来定义,这样的话就必需要首先声明游标:

SET SERVERIUTPUT ON

DECLARE

CURSOR c_emp IS SELECT ename,salary FROM emp;

R_emp c_emp%ROWTYPE;

BEGIN

OPEN c_emp;

LOOP

FETCH c_emp INTO r_emp;

EXIT WHEN c_emp%NOTFOUND;

DBMS_OUT.PUT.PUT_LINE('Salary of Employee'||r_emp.ename||'is'|| r_emp.salary);

END LOOP;

CLOSE c_emp;

END;

带参数的游标

与存储过程和函数类似,能够将参数传递给游标并在查询中使用。这对于处理在某种

条件下打开游标的状况很是有用。它的语法以下:

CURSOR cursor_name[(parameter[,parameter],...)] IS select_statement;

定义参数的语法以下:

Parameter_name [IN] data_type[{:=|DEFAULT} value]

与存储过程不一样的是,游标只能接受传递的值,而不能返回值。参数只定义数据类型

另外能够给参数设定一个缺省值,当没有参数值传递给游标时,就使用缺省值。游标

中定义的参数只是一个占位符,在别处引用该参数不必定可靠。

在打开游标时给参数赋值,语法以下:

OPEN cursor_name[value[,value]....];

参数值能够是文字或变量。

例:

DECALRE

CURSOR c_dept IS SELECT * FROM dept ORDER BY deptno;

CURSOR c_emp (p_dept VARACHAR2) IS

SELECT ename,salary

FROM emp

WHERE deptno=p_dept

ORDER BY ename

r_dept DEPT%ROWTYPE;

v_ename EMP.ENAME%TYPE;

v_salary EMP.SALARY%TYPE;

v_tot_salary EMP.SALARY%TYPE;

BEGIN

OPEN c_dept;

LOOP

FETCH c_dept INTO r_dept;

EXIT WHEN c_dept%NOTFOUND;

DBMS_OUTPUT.PUT_LINE('Department:'|| r_dept.deptno||'-'||r_dept.dname);

v_tot_salary:=0;

OPEN c_emp(r_dept.deptno);

LOOP

FETCH c_emp INTO v_ename,v_salary;

EXIT WHEN c_emp%NOTFOUND;

DBMS_OUTPUT.PUT_LINE('Name:'|| v_ename||' salary:'||v_salary);

v_tot_salary:=v_tot_salary+v_salary;

END LOOP;

CLOSE c_emp;

DBMS_OUTPUT.PUT_LINE('Toltal Salary for dept:'|| v_tot_salary);

END LOOP;

CLOSE c_dept;

END;

游标FOR循环

在大多数时候咱们在设计程序的时候都遵循下面的步骤:

一、打开游标

二、开始循环

三、从游标中取值

四、检查那一行被返回

五、处理

六、关闭循环

七、关闭游标

能够简单的把这一类代码称为游标用于循环。但还有一种循环与这种类型不相同,这

就是FOR循环,用于FOR循环的游标按照正常的声明方式声明,它的优势在于不须要显式的打开

、关闭、取数据,测试数据的存在、定义存放数据的变量等等。游标FOR 循环的语法以下:

FOR record_name IN

(corsor_name[(parameter[,parameter]...)]

| (query_difinition)

LOOP

statements

END LOOP;

下面咱们用for循环重写上面的例子:

DECALRE

CURSOR c_dept IS SELECT deptno,dname FROM dept ORDER BY deptno;

CURSOR c_emp (p_dept VARACHAR2) IS

SELECT ename,salary

FROM emp

WHERE deptno=p_dept

ORDER BY ename

v_tot_salary EMP.SALARY%TYPE;

BEGIN

FOR r_dept IN c_dept LOOP

DBMS_OUTPUT.PUT_LINE('Department:'|| r_dept.deptno||'-'||r_dept.dname);

v_tot_salary:=0;

FOR r_emp IN c_emp(r_dept.deptno) LOOP

DBMS_OUTPUT.PUT_LINE('Name:'|| v_ename||' salary:'||v_salary);

v_tot_salary:=v_tot_salary+v_salary;

END LOOP;

DBMS_OUTPUT.PUT_LINE('Toltal Salary for dept:'|| v_tot_salary);

END LOOP;

END;

在游标FOR循环中使用查询

在游标FOR循环中能够定义查询,因为没有显式声明因此游标没有名字,记录名经过

DECALRE

v_tot_salary EMP.SALARY%TYPE;

BEGIN

FOR r_dept IN (SELECT deptno,dname FROM dept ORDER BY deptno) LOOP

DBMS_OUTPUT.PUT_LINE('Department:'|| r_dept.deptno||'-'||r_dept.dname);

v_tot_salary:=0;

FOR r_emp IN (SELECT ename,salary

FROM emp

WHERE deptno=p_dept

ORDER BY ename) LOOP

DBMS_OUTPUT.PUT_LINE('Name:'|| v_ename||' salary:'||v_salary);

v_tot_salary:=v_tot_salary+v_salary;

END LOOP;

DBMS_OUTPUT.PUT_LINE('Toltal Salary for dept:'|| v_tot_salary);

END LOOP;

END;

游标中的子查询

语法以下:

CURSOR C1 IS SELECT * FROM emp

WHERE deptno NOT IN (SELECT deptno

FROM dept

WHERE dname!='ACCOUNTING');

能够看出与SQL中的子查询没有什么区别。

游标中的更新和删除

在PL/SQL中依然可使用UPDATE和DELETE语句更新或删除数据行。显式游标只有在需

要得到多行数据的状况下使用。PL/SQL提供了仅仅使用游标就能够执行删除或更新记录的方法

UPDATE或DELETE语句中的WHERE CURRENT OF子串专门处理要执行UPDATE或DELETE操做

的表中取出的最近的数据。要使用这个方法,在声明游标时必须使用FOR UPDATE子串,当对话

使用FOR UPDATE子串打开一个游标时,全部返回集中的数据行都将处于行级(ROW-LEVEL)独占

式锁定,其余对象只能查询这些数据行,不能进行UPDATE、DELETE或SELECT...FOR UPDATE操

语法:

FOR UPDATE [OF [schema.]table.column[,[schema.]table.column]..

[nowait]

在多表查询中,使用OF子句来锁定特定的表,若是忽略了OF子句,那么全部表中选择

的数据行都将被锁定。若是这些数据行已经被其余会话锁定,那么正常状况下ORACLE将等待,

在UPDATE和DELETE中使用WHERE CURRENT OF子串的语法以下:

WHERE{CURRENT OF cursor_name|search_condition}

例:

DELCARE

CURSOR c1 IS SELECT empno,salary

FROM emp

WHERE comm IS NULL

FOR UPDATE OF comm;

v_comm NUMBER(10,2);

BEGIN

FOR r1 IN c1 LOOP

IF r1.salary<500 THEN

v_comm:=r1.salary*0.25;   ELSEIF r1.salary<1000 THEN

v_comm:=r1.salary*0.20;   ELSEIF r1.salary<3000 THEN

v_comm:=r1.salary*0.15;

ELSE

v_comm:=r1.salary*0.12;

END IF;

UPDATE emp;

SET comm=v_comm

WHERE CURRENT OF c1l;

END LOOP;

END

附[WHERE CURRENT OF ]:

where   current   of   子句只能在定义游标的时候使用了   for   update语句才可使用。

for   v_sor   in   t_sor   loop

update   aa   set   value='3'   where   current   of   v_sor;       --已限制了条件了,更新只是当前记录集

end   loop;

上面的语句中   v_sor   是   t_sor中的一行数据,更新的时候应该用   current  of   t_sor吧,由于v_sor只是一个隐式游标,它自己不是经过   for   update定义的,只是表明了循环中t_sor的当前记录。这样,你的update语句或者delete语句的做用范围就只在你循环的当前行的范围中了。

要注意的是,用for   update定义的游标会让数据库对涉及的行加锁,别的会话若是要访问该游标中的行便会进入等待状态。你也能够明确指明要加锁的列,用   for   update   of   "列名"     就能够了。

若是你的select   for   update选中的行已经被别的会话加锁的话,会话就须要等待解锁,若是别的会话一直不解锁,那么你的select就会一直等待下去,若是你不想等,只需在for   update后面加上nowait就能够解决这个问题了,这样你的选择会当即返回。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值