版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
复习复习1
动态SQL包括本地动态SQL和DBMS_SQL动态SQL两种实现方法。
动态SQL简介动态SQL包括本地动态SQL和DBMS_SQL动态2为了在动态SQL中处理DDL、DCL、DML以及单行selectinto语句,需要使用executeimmediate语句。Executeimmediate语句为了在动态SQL中处理DDL、DCL、DML以及单行s3语法如下:
executeimmediatedyn_string1、executeimmediate语句处理DDL和DCL语句注:
dyn_string:用于指定存放DDL或DCL文本的字符串变量。语法如下:1、executeimmediate语句处理DD4(1)处理无占位符和returning子句的DML语句语法如下:
executeimmediatedyn_string2、使用executeimmediate语句处理DML语句注:dyn_string:用于指定存放DML文本的字符串变量。(1)处理无占位符和returning子句的DML语句2、使5(2)处理包含占位符的DML语句注:
1、占位符必须以冒号开始,可以使用任何名称。
2、bind_argument用于为占位符提供数据。语法如下:
executeimmediatedyn_string
usingbind_argument[,……](2)处理包含占位符的DML语句注:1、占位符必须以冒号开6(3)处理包含returning子句的DML语句注:
1、当使用returninginto子句接收数据时,只能接受DML语句返回的单行数据。
2、当在动态DML语句中指定returninginto子句时,需要在into后使用占位符。语法如下:
executeimmediatedyn_string
returningintovar1[,……](3)处理包含returning子句的DML语句注:17语法如下:
executeimmediatedyn_string
intovar1,……3、使用executeimmediate语句处理单行语句语法如下:3、使用executeimmediate语句处理8游标变量是基于REFCURSOR类型所定义的变量。使用显示游标只能定义静态游标,而通过使用游标变量可以在打开游标时指定游标所对应的select语句,从而实现动态游标。游标变量游标变量是基于REFCURSOR类型所定义的变量。使9游标变量包括定义游标变量、打开游标变量、提取数据和关闭游标四个阶段。游标变量1、定义REFCURSOR类型和游标变量语法如下:
TYPEref_type_nameISREFCURSOR[RETURNreturn_type];cursor_variableref_type_name;游标变量包括定义游标变量、打开游标变量、提取数据和关闭102、打开游标变量语法如下:
OPENcursor_variableFORselect……3、提取数据语法如下:
FETCHcursor_variableINTOvar1,……4、关闭游标变量语法如下:
CLOSEcursor_variable2、打开游标变量语法如下:3、提取数据语法如下:4、关闭游标11游标NEW游标NEW12为什么要使用游标?关系数据库管理系统实质是面向集合的,在Oracle中并没有一种描述表中单一记录的表达形式,除非使用where子句来限制只有一条记录被选中。游标允许应用程序对查询语句返回的行结果集中的每一行进行相同或不同的操作,而不是一次对整个结果集进行同一种操作。为什么要使用游标?关系数据库管理系统实质是面向集13什么是游标?就本质而言,游标实际上是一种能从包括多条数据记录的结果集中每次提取一条记录的机制。什么是游标?就本质而言,游标实际上是一14游标由结果集和结果集中指向特定记录的游标位置组成,游标充当指针的作用。尽管游标能够遍历查询结果中的所有行,但它一次只能指向一行。游标的构成游标由结果集和结果集中指向特定记录的游标位15Oracle中游标的分类在Oracle中游标可以分为两种类型:显式游标和隐式游标。
显式游标:用于处理select语句返回的多行数据。隐式游标:用于处理单行selectinto语句和DML语句。两者区别(p129)Oracle中游标的分类在Oracle中游标可以分为两16显式游标的使用使用显式游标分四个步骤:
2.打开游标
OPENcursor_name1.声明游标
CURSORcursor_nameISselect_statement3.从游标中查找信息
FETCHcursor_nameINTOvar1,var2,……4.关闭游标
CLOSEcursor_name显式游标的使用使用显式游标分四个步骤:2.打开游17例:声明游标,对应的集合是表emp中所有的员工姓名。DECLAREcursorc_emp_enameisselectenamefromscott.emp;显式游标的使用例:声明游标,对应的集合是表emp中所有的员工姓名。DEC18例:显示游标c_emp_ename中的第一行信息。DECLARE
cursorc_emp_enameisselectenamefromscott.emp;v_enamescott.emp.ename%type;BEGIN
openc_emp_ename;
fetchc_emp_enameintov_ename;dbms_output.put_line(v_ename);
closec_emp_ename;END;显式游标的使用思考:如何显示游标c_emp_ename中的所有信息?例:显示游标c_emp_ename中的第一行信息。DECLA19练习:显示游标c_emp_ename中的所有信息。DECLARE
cursorc_emp_enameisselectenamefromscott.emp;v_enameemp.ename%type;v_countbinary_integer;BEGINselectcount(rowid)intov_countfromscott.emp;
openc_emp_ename;
foriin1..v_countloop
fetchc_emp_enameintov_ename;dbms_output.put_line(v_ename);
endloop;
closec_emp_ename;END;显式游标的使用思考:如何获取指定职工编号的员工姓名?练习:显示游标c_emp_ename中的所有信息。DECLA20参数游标参数游标是指带有参数的游标。2.打开游标
OPENcursor_name(参数值)1.声明参数游标CURSORcursor_name(参数名参数数据类型,..)
ISselect…from….where……注:1、定义参数只能指定数据类型,不能指定长度。2、必须在游标select语句的where子句中引用游标参数,否则失去了定义参数游标的意义。参数游标参数游标是指带有参数的游标。2.打开游标1.声21DECLARE
cursorc_emp(v_empnonumber)
is
select*fromscott.empwhereempno=v_empno;v_empscott.emp%rowtype;BEGIN
openc_emp(7369);fetchc_empintov_emp;dbms_output.put_line(v_emp.ename);closec_emp;END;例:使用参数游标获取7369员工的所有信息。思考:使用游标获取任何编号员工的所有信息?openc_emp(&number)DECLARE例:使用参数游标获取7369员工的所有信息。思22
DECLAREcursorc_empisselect*fromscott.empwhereempno=&v_empno;v_empscott.emp%rowtype;BEGINopenc_emp;fetchc_empintov_emp;dbms_output.put_line(v_emp.ename);END;练习:使用非参数游标(替代变量)获取任何编号员工的所有信息。思考:如何处理编号不存在的情况?DECLARE练习:使用非参数游标(替代变量)获取任何编号23显式游标属性
当使用显示游标时,需要使用游标属性确定显示游标的执行信息,显示游标包括%isopen、%found、%notfound、%rowcount四种属性。注:当引用显示游标属性时,需要带有游标名作为前缀(例:emp_cursor%rowcount)显式游标属性当使用显示游标时,需要使用游标24显式游标属性
%found是一个布尔属性。如果前一个fetch语句返一个行,那么它返回true,否则返回false。如果在相关联的游标还没有打开进行引用,那么会返回错误。%notfound的属性与%found相反。%isopen也是布尔属性,用来确定相关的游标是否被打开。%rowcount此数字属性返回目前为止由游标返回行的数目,即fetch语句后的得到数字。如果在相关联的游标还没有打开或者已经关闭的时候进行引用,会返回错误。显式游标属性%found是一个布尔属性。如果前一个fetc25
DECLAREcursorc_empisselect*fromscott.empwhereempno=&v_empno;v_empscott.emp%rowtype;BEGINopenc_emp;fetchc_empintov_emp;
IFc_emp%foundthendbms_output.put_line(v_emp.ename);ELSEdbms_output.put_line('您输入的编号不存在!');ENDIF;END;例:使用游标获取任何编号员工的所有信息,对于不存在的编号显示“您输入的编号不存在”。DECLARE例:使用游标获取任何编号员工的所有信息,对于26练习:使用参数游标获取任何部门编号的员工姓名信息,并返回记录条数。对于不存在的编号显示“您输入的编号不存在”。练习:使用参数游标获取任何部门编号的员工姓名信息,并返回记录27
DECLAREcursorc_e_dept(dnonumber)isselect*fromscott.empwheredeptno=dno;v_emp_deptscott.emp%rowtype;v_dnoemp.deptno%type:=&deno;v_countbinary_integer;BEGINselectcount(rowid)intov_countfromempwheredeptno=v_dno;openc_e_dept(v_dno);
fetchc_e_deptintov_emp_dept;IFc_e_dept%foundthenforiin1..v_countloopdbms_output.put_line(v_emp_dept.ename);fetchc_e_deptintov_emp_dept;endloop;dbms_output.put_line('共有员工人数:'||c_e_dept%rowcount);ELSEdbms_output.put_line('您输入的编号不存在!');ENDIF;closec_e_dept;END;必须先获取数据才能有found的返回值。DECLARE必须先获取数据才能有found的返回值。28
cursorfor循环forrecord_namein(corsor_name[(参数1[,参数2]...)]loopstatementsendloop;
为了简化游标处理,可以使用游标for循环,当使用游标for循环时,Oracle会隐含的打开游标、提取数据并关闭游标。注:
record_name是基于游标隐含定义的记录变量。在执行循环之前,Oracle会隐含的打开游标,并且每循环一次自动提取一行数据,当所有数据提取后自动退出循环并关闭游标。cursorfor循环forrecord_namei29
例:编写PL/SQL块,定义游标,使用替代变量输入部门号,并以工资降序显示该部门所有雇员的名称和工资。DECLARE
cursoremp_cursorisselectename,salfromscott.empwheredeptno=&noorderbysaldesc;BEGIN
foremp_recordinemp_cursorloopdbms_output.put_line('姓名:'||emp_record.ename||',工资:'||emp_record.sal);
endloop;END;例:编写PL/SQL块,定义游标,使用替代变量输入部门号,30练习:编写PL/SQL块,定义参数游标(参数:岗位),使用替代变量输入岗位,并显示该岗位所有雇员的名称和工资。DECLARE
cursoremp_cursor(titlevarchar2)
isselectename,salfromscott.empwherejob=title;BEGIN
foremp_recordin
emp_cursor('&job')
loopdbms_output.put_line('姓名:'||emp_record.ename||',工资:'||emp_record.sal);
endloop;END;练习:编写PL/SQL块,定义参数游标(参数:岗位),使用替31分析:DECLAREcursorc_e_dept(dnonumber)isselect*fromscott.empwheredeptno=dno;BEGINforemp_recordinc_e_dept(&dno)loopdbms_output.put_line(emp_record.ename);IFc_e_dept%rowcount>0thendbms_output.put_line('共有员工人数:'||c_e_dept%rowcount);ELSEdbms_output.put_line('您输入的编号不存在!');ENDIF;endloop;END;练习:使用参数游标获取任何部门编号的员工姓名信息,并返回记录条数。对于不存在的编号显示“您输入的编号不存在”。分析:练习:使用参数游标获取任何部门编号的员工姓名信息,并返32更新或删除游标行通过使用显式游标,不仅可以取得游标结果集的数据,而且可以更新或删除游标结果集的当前行。DELETEFROMtable_nameWHERECURRENTOFcursor_name;CURSORcursor_nameISselect_statementFORUPDATE;UPDATEtable_nameSET……..WHERECURRENTOFcursor_name;更新或删除游标行通过使用显式游标,不仅可以取得游标结果集的数33例:要求输入部门号,输出该部门所有雇员姓名及原工资,并将雇员的工资提高10%。DECLARECURSORemp_cursorISSELECTename,sal,deptnoFROMscott.empWHEREdeptno=&noFORUPDATE;BEGINFORemp_recordINemp_cursorLOOP dbms_output.put_line('姓名:'||emp_record.ename ||',原工资:'||emp_record.sal);
UPDATEscott.empSETsal=sal*1.1WHERECURRENTOFemp_cursor;ENDLOOP;END;例:要求输入部门号,输出该部门所有雇员姓名及原工资,并将雇员34练习:要求输入雇员号,输出该雇员姓名及原工资,并将雇员删除。
DECLARECURSORemp_cursorISSELECTename,salFROMscott.empWHEREempno=&noFORupdate;BEGINFORemp_recordINemp_cursorLOOPdbms_output.put_line('姓名:'||emp_record.ename||',原工资:'||emp_record.sal);DELETEscott.empWHERECURRENTOFemp_cursor;ENDLOOP;END;练习:要求输入雇员号,输出该雇员姓名及原工资,并将雇员删除。35隐式游标属性
隐式游标也称SQL游标,用于处理单行selectinto语句和DML语句。一个SQL游标不能使用open、fetch、close来操作。与显式游标相同,SQL游标也有%isopen、%found、%notfound、%rowcount四种属性。具体看书。注:当引用隐式游标属性时,需要带有SQL作为前缀(例:SQL%rowcount)隐式游标属性隐式游标也称SQL游标,用于处理单行selec36书例6.11declaretypeename_typeistableofemp.ename%typenotnullindexbybinary_integer;typedname_typeistableofdept.dname%typeindexbybinary_integer;enamelistename_type;dnamelistdname_type;subscriptbinary_integer:=1;beginforvarin(selectename,dnamefromemp,deptwhereemp.deptno=dept.deptno)loopenamelist(subscript):=var.ename;dnamelist(subscript):=var.dname;dbms_output.put_line('employeenameis'||enamelist(subscript)||',departmentnameis'||dnamelist(subscript));subscript:=subscript+1;endloop;dbms_output.put_line('totalnumberofemployeesare'||subscript);end;/cursorfor循环书例6.11declarecursorfor循环37书例6.12SQL%found属性declarev_empnoemp.empno%type:=&v_empno;v_empemp%rowtype;beginselect*intov_empfromempwhereempno=v_empno;ifSQL%foundthendbms_output.put_line(v_emp.empno);dbms_output.put_line(v_emp.ename);endif;end;/书例6.12SQL%found属性declare38书例6.13SQL%notfound属性declarev_empnoemp.empno%type:=7000;beginupdateemp1setename='sem108'whereempno=v_empno;ifSQL%foundthendbms_output.put_line('DeletedOK!');endif;ifSQL%notfoundthendbms_output.put_line('employeenumber:'||v_empno||'doesnotexists!');endif;end;/书例6.13SQL%notfound属性declare39书例6.14SQL%isopen属性SQL%rowcount属性declarev_deptnoemp.deptno%type:=10;begindeleteemp1wheredeptno=v_deptno;ifSQL%foundthendbms_output.put_line(SQL%rowcount||'deletedOK!');endif;ifSQL%notfoundthendbms_output.put_line('deptnonumber:'||v_deptno||'doesnotexists!');endif;end;/书例6.14SQL%isopen属性declare40练习:1、当使用显示游标时,在执行了哪条语句后应该检查游标是否包含行?
A、openB、fetchC、closeD、cursor2、在以下哪些语句中可以包含wherecurrentof子句?
A、openB、fetchC、delete
D、selectE、updateF、cursor√√√练习:1、当使用显示游标时,在执行了哪条语句后应该检查游标是41练习:3、请查看以下游标定义语句,哪行会引起错误?DECLARECURSORcust_cursor(p_cust_id,p_last_name)ISSELECTcust_id,first_name,last_nameFROMcustomer
WHEREcust_id=p_cust_idANDlast_name=p_last_name;
定义游标变量需指明变量的数据类型A、2B、3C、4D、5E、64、课后习题第二、三题(课本149页)√练习:3、请查看以下游标定义语句,哪行会引起错误?4、课后习42小结游标概念游标分类显示游标的使用参数游标游标属性Cursorfor循环更新、删除游标行隐式游标的使用小结游标概念43异常处理异常处理44
为了提高应用程序的健壮性,使得应用程序可以安全正常的运行,应用开发人员应该考虑到PL/SQL块可能出现的各种异常情况,并进行相应的处理。通常使用异常(Exception),可以处理PL/SQL块的各种异常情况。引言为了提高应用程序的健壮性,使得应用程序可以安全正常45
异常(Exception)是一种PL/SQL标识符,它包括预定义异常、非预定义异常、自定义异常。异常简介
当编写PL/SQL块时,应该捕捉并处理各种可能出现的异常。如果不捕捉和处理异常,那么Oracle会将错误传递到调用环境;如果捕捉并处理异常,那么Oracle会在PL/SQL块内解决运行错误。异常(Exception)是一种PL/46DECLAREv_enameemp.ename%TYPE;BEGINSELECTenameINTOv_enameFROMscott.empWHEREempno=&no;dbms_output.put_line('雇员名:'||v_ename);END;未捕捉异常,传递错误到调用环境DECLARE未捕捉异常,传递错误到调用环境47DECLAREv_enameemp.ename%TYPE;BEGINSELECTenameINTOv_enameFROMempWHEREempno=&no;dbms_output.put_line('雇员名:'||v_ename);EXCEPTIONWHENNO_DATA_FOUNDTHENdbms_output.put_line('雇员号不正确,请核实雇员号!');END;捕捉异常并处理DECLARE捕捉异常并处理48异常处理部分的语法如下:
BEGIN……EXCEPTION
whenexception_namethen
codeforhandingexception_name
[whenanother_exceptionthen
codeforhandinganother_exception]
[whenothersthen
codeforhandinganyotherexception.]END;异常处理部分的语法如下:BEGIN49预定义异常是指PL/SQL所提供的系统异常。Oracle为应用开发人员提供了21个预定义异常,每个预定义异常对应一个特定的Oracle错误,当PL/SQL块出现这些Oracle错误时,会隐含的处罚相应的预定义异常。一、预定义异常应用预定义异常预定义异常是指PL/SQL所提供的系50
非预定义异常用于处理与预定义异常无关的Oracle错误。预定义异常只能用于处理21种Oracle错误,而PL/SQL块还会遭遇其他Oracle错误(例如:完整性约束ORA-02291)二、非预定义异常例:beginupdatescott.empsetdeptno=&nowhereempno=&eno;end;非预定义异常用于处理与预定义异常无关的Oracle错511.在declare部分定义异常标识符。2.在declare部分建立Oracle错误号与异常之间的关联。3.捕捉并处理异常。使用非预定义异常的步骤:语法如下:declareexception_nameexception;
pragma
exception_init(exception_name,-2291);…………….1.在declare部分定义异常标识符。使用非预定义异常的步52例如:DECLARE
e_integrityEXCEPTION;PRAGMAEXCEPTION_INIT(e_integrity,-2291);namescott.emp.ename%TYPE:=LOWER('&name');dnoscott.emp.deptno%TYPE:=&dno;BEGINUPDATEscott.empSETdeptno=dnoWHERELOWER(ename)=name;EXCEPTIONWHENe_integrityTHENdbms_output.put_line('该部门不存在');END;例如:53
为了在PL/SQL块处理Oracle错误,应用开发人员可以使用预定义异常或者非预定义异常。
注:当执行内嵌update和delete语句时,如果没有更新或删除行,那么PL/SQL块不会给出任何提示信息,为了使得用户可以获得更有用,更有意义的消息,需要使用自定义异常。三、自定义异常为了在PL/SQL块处理Oracle错误,应用开发人541.在declare部分定义异常标识符。2.在执行部分使用raise语句显示触发异常。3.捕捉并处理异常。使用自定义异常步骤:语法:declare
exception_nameexception;begin…………….
raiseexception_name;
exception……..end;1.在declare部分定义异常标识符。使用自定义异常步骤:55例如:DECLARE
e_integrityEXCEPTION; e_no_rowsEXCEPTION; PRAGMAEXCEPTION_INIT(e_integrity,-2291); namescott.emp.ename%TYPE:=LOWER('&name'); dnoscott.emp.deptno%TYPE:=&dno;BEGIN UPDATEscott.empSETdeptno=dnoWHERELOWER(ename)=name;
IFSQL%NOTFOUNDTHEN RAISEe_no_rows; ENDIF;EXCEPTION WHENe_integrityTHEN dbms_output.put_line('该部门不存在'); WHENe_no_rowsTHEN dbms_output.put_line('该雇员不存在');END;例如:56练习:1、RAISE语句应该放在PL/SQL块的哪个部分?
A、定义部分B、执行部分C、异常处理部分2、当在异常和Oracle错误之间建立关联时,应该在哪个部分完成?A、定义部分B、执行部分C、异常处理部分3、假定在emp表上定义了check约束要求雇员的工资不能高于6000,为了处理工资超过6000可能出现的错误,应该使用哪种异常?A、预定义异常B、非预定义异常C、自定义异常√√√练习:1、RAISE语句应该放在PL/SQL块的哪个部分?257练习:4、编写PL/SQL块,使用替代变量输入雇员号,显示雇员名及其工资。如果雇员不存在,则显示“请输入正确的雇员号”5、编写PL/SQL块,使用替代变量输入雇员号和新部门号,更新雇员部门,如果部门不存在,则显示“请输入正确的部门号!”;如果雇员不存在,则显示“请输入正确的雇员号”练习:4、编写PL/SQL块,使用替代变量输入雇员号,显示雇58
在一个异常中,SQLCODE返回Oracle错误的序号,而SQLERRM返回的是相应的错误消息,错误消息首先显示是错误代码。SQLCODE和SQLERRM注:1.SQLCODE通常返回的是负数。对于用户自定义的异常,SQLCODE返回的是+1,而SQLERRM返回的是自定义错误信息。2.如果没有异常被触发,则SQLCODE返回0,SQLERRM返回ORA-0000:normal,successfulcompletion。在一个异常中,SQLCODE返回Oracle错误的序号,59declarev_empscott.emp%rowtype;empno_out_of_rangeexception;beginselect*intov_empfromscott.emp;ifSQL%foundthendbms_output.put_line(v_emp.empno);dbms_output.put_line(v_emp.ename);endif;exceptionwhenempno_out_of_rangethendbms_output.put_line('empnoisoutofrange');whenno_data_foundthendbms_output.put_line('notfoundrecord');whentoo_many_rowsthendbms_output.put_line('toomanyrecord');
dbms_output.put_line(SQLCode||','||SQLERRM);end;/declare60复习复习61
动态SQL包括本地动态SQL和DBMS_SQL动态SQL两种实现方法。
动态SQL简介动态SQL包括本地动态SQL和DBMS_SQL动态62为了在动态SQL中处理DDL、DCL、DML以及单行selectinto语句,需要使用executeimmediate语句。Executeimmediate语句为了在动态SQL中处理DDL、DCL、DML以及单行s63语法如下:
executeimmediatedyn_string1、executeimmediate语句处理DDL和DCL语句注:
dyn_string:用于指定存放DDL或DCL文本的字符串变量。语法如下:1、executeimmediate语句处理DD64(1)处理无占位符和returning子句的DML语句语法如下:
executeimmediatedyn_string2、使用executeimmediate语句处理DML语句注:dyn_string:用于指定存放DML文本的字符串变量。(1)处理无占位符和returning子句的DML语句2、使65(2)处理包含占位符的DML语句注:
1、占位符必须以冒号开始,可以使用任何名称。
2、bind_argument用于为占位符提供数据。语法如下:
executeimmediatedyn_string
usingbind_argument[,……](2)处理包含占位符的DML语句注:1、占位符必须以冒号开66(3)处理包含returning子句的DML语句注:
1、当使用returninginto子句接收数据时,只能接受DML语句返回的单行数据。
2、当在动态DML语句中指定returninginto子句时,需要在into后使用占位符。语法如下:
executeimmediatedyn_string
returningintovar1[,……](3)处理包含returning子句的DML语句注:167语法如下:
executeimmediatedyn_string
intovar1,……3、使用executeimmediate语句处理单行语句语法如下:3、使用executeimmediate语句处理68游标变量是基于REFCURSOR类型所定义的变量。使用显示游标只能定义静态游标,而通过使用游标变量可以在打开游标时指定游标所对应的select语句,从而实现动态游标。游标变量游标变量是基于REFCURSOR类型所定义的变量。使69游标变量包括定义游标变量、打开游标变量、提取数据和关闭游标四个阶段。游标变量1、定义REFCURSOR类型和游标变量语法如下:
TYPEref_type_nameISREFCURSOR[RETURNreturn_type];cursor_variableref_type_name;游标变量包括定义游标变量、打开游标变量、提取数据和关闭702、打开游标变量语法如下:
OPENcursor_variableFORselect……3、提取数据语法如下:
FETCHcursor_variableINTOvar1,……4、关闭游标变量语法如下:
CLOSEcursor_variable2、打开游标变量语法如下:3、提取数据语法如下:4、关闭游标71游标NEW游标NEW72为什么要使用游标?关系数据库管理系统实质是面向集合的,在Oracle中并没有一种描述表中单一记录的表达形式,除非使用where子句来限制只有一条记录被选中。游标允许应用程序对查询语句返回的行结果集中的每一行进行相同或不同的操作,而不是一次对整个结果集进行同一种操作。为什么要使用游标?关系数据库管理系统实质是面向集73什么是游标?就本质而言,游标实际上是一种能从包括多条数据记录的结果集中每次提取一条记录的机制。什么是游标?就本质而言,游标实际上是一74游标由结果集和结果集中指向特定记录的游标位置组成,游标充当指针的作用。尽管游标能够遍历查询结果中的所有行,但它一次只能指向一行。游标的构成游标由结果集和结果集中指向特定记录的游标位75Oracle中游标的分类在Oracle中游标可以分为两种类型:显式游标和隐式游标。
显式游标:用于处理select语句返回的多行数据。隐式游标:用于处理单行selectinto语句和DML语句。两者区别(p129)Oracle中游标的分类在Oracle中游标可以分为两76显式游标的使用使用显式游标分四个步骤:
2.打开游标
OPENcursor_name1.声明游标
CURSORcursor_nameISselect_statement3.从游标中查找信息
FETCHcursor_nameINTOvar1,var2,……4.关闭游标
CLOSEcursor_name显式游标的使用使用显式游标分四个步骤:2.打开游77例:声明游标,对应的集合是表emp中所有的员工姓名。DECLAREcursorc_emp_enameisselectenamefromscott.emp;显式游标的使用例:声明游标,对应的集合是表emp中所有的员工姓名。DEC78例:显示游标c_emp_ename中的第一行信息。DECLARE
cursorc_emp_enameisselectenamefromscott.emp;v_enamescott.emp.ename%type;BEGIN
openc_emp_ename;
fetchc_emp_enameintov_ename;dbms_output.put_line(v_ename);
closec_emp_ename;END;显式游标的使用思考:如何显示游标c_emp_ename中的所有信息?例:显示游标c_emp_ename中的第一行信息。DECLA79练习:显示游标c_emp_ename中的所有信息。DECLARE
cursorc_emp_enameisselectenamefromscott.emp;v_enameemp.ename%type;v_countbinary_integer;BEGINselectcount(rowid)intov_countfromscott.emp;
openc_emp_ename;
foriin1..v_countloop
fetchc_emp_enameintov_ename;dbms_output.put_line(v_ename);
endloop;
closec_emp_ename;END;显式游标的使用思考:如何获取指定职工编号的员工姓名?练习:显示游标c_emp_ename中的所有信息。DECLA80参数游标参数游标是指带有参数的游标。2.打开游标
OPENcursor_name(参数值)1.声明参数游标CURSORcursor_name(参数名参数数据类型,..)
ISselect…from….where……注:1、定义参数只能指定数据类型,不能指定长度。2、必须在游标select语句的where子句中引用游标参数,否则失去了定义参数游标的意义。参数游标参数游标是指带有参数的游标。2.打开游标1.声81DECLARE
cursorc_emp(v_empnonumber)
is
select*fromscott.empwhereempno=v_empno;v_empscott.emp%rowtype;BEGIN
openc_emp(7369);fetchc_empintov_emp;dbms_output.put_line(v_emp.ename);closec_emp;END;例:使用参数游标获取7369员工的所有信息。思考:使用游标获取任何编号员工的所有信息?openc_emp(&number)DECLARE例:使用参数游标获取7369员工的所有信息。思82
DECLAREcursorc_empisselect*fromscott.empwhereempno=&v_empno;v_empscott.emp%rowtype;BEGINopenc_emp;fetchc_empintov_emp;dbms_output.put_line(v_emp.ename);END;练习:使用非参数游标(替代变量)获取任何编号员工的所有信息。思考:如何处理编号不存在的情况?DECLARE练习:使用非参数游标(替代变量)获取任何编号83显式游标属性
当使用显示游标时,需要使用游标属性确定显示游标的执行信息,显示游标包括%isopen、%found、%notfound、%rowcount四种属性。注:当引用显示游标属性时,需要带有游标名作为前缀(例:emp_cursor%rowcount)显式游标属性当使用显示游标时,需要使用游标84显式游标属性
%found是一个布尔属性。如果前一个fetch语句返一个行,那么它返回true,否则返回false。如果在相关联的游标还没有打开进行引用,那么会返回错误。%notfound的属性与%found相反。%isopen也是布尔属性,用来确定相关的游标是否被打开。%rowcount此数字属性返回目前为止由游标返回行的数目,即fetch语句后的得到数字。如果在相关联的游标还没有打开或者已经关闭的时候进行引用,会返回错误。显式游标属性%found是一个布尔属性。如果前一个fetc85
DECLAREcursorc_empisselect*fromscott.empwhereempno=&v_empno;v_empscott.emp%rowtype;BEGINopenc_emp;fetchc_empintov_emp;
IFc_emp%foundthendbms_output.put_line(v_emp.ename);ELSEdbms_output.put_line('您输入的编号不存在!');ENDIF;END;例:使用游标获取任何编号员工的所有信息,对于不存在的编号显示“您输入的编号不存在”。DECLARE例:使用游标获取任何编号员工的所有信息,对于86练习:使用参数游标获取任何部门编号的员工姓名信息,并返回记录条数。对于不存在的编号显示“您输入的编号不存在”。练习:使用参数游标获取任何部门编号的员工姓名信息,并返回记录87
DECLAREcursorc_e_dept(dnonumber)isselect*fromscott.empwheredeptno=dno;v_emp_deptscott.emp%rowtype;v_dnoemp.deptno%type:=&deno;v_countbinary_integer;BEGINselectcount(rowid)intov_countfromempwheredeptno=v_dno;openc_e_dept(v_dno);
fetchc_e_deptintov_emp_dept;IFc_e_dept%foundthenforiin1..v_countloopdbms_output.put_line(v_emp_dept.ename);fetchc_e_deptintov_emp_dept;endloop;dbms_output.put_line('共有员工人数:'||c_e_dept%rowcount);ELSEdbms_output.put_line('您输入的编号不存在!');ENDIF;closec_e_dept;END;必须先获取数据才能有found的返回值。DECLARE必须先获取数据才能有found的返回值。88
cursorfor循环forrecord_namein(corsor_name[(参数1[,参数2]...)]loopstatementsendloop;
为了简化游标处理,可以使用游标for循环,当使用游标for循环时,Oracle会隐含的打开游标、提取数据并关闭游标。注:
record_name是基于游标隐含定义的记录变量。在执行循环之前,Oracle会隐含的打开游标,并且每循环一次自动提取一行数据,当所有数据提取后自动退出循环并关闭游标。cursorfor循环forrecord_namei89
例:编写PL/SQL块,定义游标,使用替代变量输入部门号,并以工资降序显示该部门所有雇员的名称和工资。DECLARE
cursoremp_cursorisselectename,salfromscott.empwheredeptno=&noorderbysaldesc;BEGIN
foremp_recordinemp_cursorloopdbms_output.put_line('姓名:'||emp_record.ename||',工资:'||emp_record.sal);
endloop;END;例:编写PL/SQL块,定义游标,使用替代变量输入部门号,90练习:编写PL/SQL块,定义参数游标(参数:岗位),使用替代变量输入岗位,并显示该岗位所有雇员的名称和工资。DECLARE
cursoremp_cursor(titlevarchar2)
isselectename,salfromscott.empwherejob=title;BEGIN
foremp_recordin
emp_cursor('&job')
loopdbms_output.put_line('姓名:'||emp_record.ename||',工资:'||emp_record.sal);
endloop;END;练习:编写PL/SQL块,定义参数游标(参数:岗位),使用替91分析:DECLAREcursorc_e_dept(dnonumber)isselect*fromscott.empwheredeptno=dno;BEGINforemp_recordinc_e_dept(&dno)loopdbms_output.put_line(emp_record.ename);IFc_e_dept%rowcount>0thendbms_output.put_line('共有员工人数:'||c_e_dept%rowcount);ELSEdbms_output.put_line('您输入的编号不存在!');ENDIF;endloop;END;练习:使用参数游标获取任何部门编号的员工姓名信息,并返回记录条数。对于不存在的编号显示“您输入的编号不存在”。分析:练习:使用参数游标获取任何部门编号的员工姓名信息,并返92更新或删除游标行通过使用显式游标,不仅可以取得游标结果集的数据,而且可以更新或删除游标结果集的当前行。DELETEFROMtable_nameWHERECURRENTOFcursor_name;CURSORcursor_nameISselect_statementFORUPDATE;UPDATEtable_nameSET……..WHERECURRENTOFcursor_name;更新或删除游标行通过使用显式游标,不仅可以取得游标结果集的数93例:要求输入部门号,输出该部门所有雇员姓名及原工资,并将雇员的工资提高10%。DECLARECURSORemp_cursorISSELECTename,sal,deptnoFROMscott.empWHEREdeptno=&noFORUPDATE;BEGINFORemp_recordINemp_cursorLOOP dbms_output.put_line('姓名:'||emp_record.ename ||',原工资:'||emp_record.sal);
UPDATEscott.empSETsal=sal*1.1WHERECURRENTOFemp_cursor;ENDLOOP;END;例:要求输入部门号,输出该部门所有雇员姓名及原工资,并将雇员94练习:要求输入雇员号,输出该雇员姓名及原工资,并将雇员删除。
DECLARECURSORemp_cursorISSELECTename,salFROMscott.empWHEREempno=&noFORupdate;BEGINFORemp_recordINemp_cursorLOOPdbms_output.put_line('姓名:'||emp_record.ename||',原工资:'||emp_record.sal);DELETEscott.empWHERECURRENTOFemp_cursor;ENDLOOP;END;练习:要求输入雇员号,输出该雇员姓名及原工资,并将雇员删除。95隐式游标属性
隐式游标也称SQL游标,用于处理单行selectinto语句和DML语句。一个SQL游标不能使用open、fetch、close来操作。与显式游标相同,SQL游标也有%isopen、%found、%notfound、%rowcount四种属性。具体看书。注:当引用隐式游标属性时,需要带有SQL作为前缀(例:SQL%rowcount)隐式游标属性隐式游标也称SQL游标,用于处理单行selec96书例6.11declaretypeename_typeistableofemp.ename%typenotnullindexbybinary_integer;typedname_typeistableofdept.dname%typeindexbybinary_integer;enamelistename_type;dnamelistdname_type;subscriptbinary_integer:=1;beginforvarin(selectename,dnamefromemp,deptwhereemp.deptno=dept.deptno)loopenamelist(subscript):=var.ename;dnamelist(subscript):=var.dname;dbms_output.put_line('employeenameis'||enamelist(subscript)||',departmentnameis'||dnamelist(subscript));subscript:=subscript+1;endloop;dbms_output.put_line('totalnumberofemployeesare'||subscript);end;/cursorfor循环书例6.11declarecursorfor循环97书例6.12SQL%found属性declarev_empnoemp.empno%type:=&v_empno;v_empemp%rowtype;beginselect*intov_empfromempwhereempno=v_empno;ifSQL%foundthendbms_output.put_line(v_emp.empno);dbms_output.put_line(v_emp.ename);endif;end;/书例6.12SQL%found属性declare98书例6.13SQL%notfound属性declarev_empnoemp.empno%type:=7000;beginupdateemp1setename='sem108'whereempno=v_empno;ifSQL%foundthendbms_output.put_line('DeletedOK!');endif;ifSQL%notfoundthendbms_output.put_line('employeenumber:'||v_empno||'doesnotexists!');endif;end;/书例6.13SQL%notfound属性declare99书例6.14SQL%isopen属性SQL%rowcount属性declarev_deptnoemp.deptno%type:=10;begindeleteemp1wheredeptno=v_deptno;ifSQL%foundthendbms_output.put_line(SQL%rowcount||'deletedOK!');endif;ifSQL%notfoundthendbms_output.put_line('deptnonumber:'||v_deptno||'doesnotexists!');endif;end;/书例6.14SQL%isopen属性declare100练习:1、当使用显示游标时,在执行了哪条语句后应该检查游标是否包含行?
A、open
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 单位管理制度呈现合集【人力资源管理篇】
- 2024年厂年度劳动竞赛的工作总结
- 《广告的社会功能》课件
- 第1单元 中华人民共和国的成立与巩固 (B卷·能力提升练)(解析版)
- 《孟子生平简介》课件
- 《杜绝校园欺凌》课件
- 超市客服话务员工作总结
- 探索生态之谜
- 2023年项目安全培训考试题(能力提升)
- 2023年项目部治理人员安全培训考试题附完整答案(必刷)
- 2024年7月国家开放大学法律事务专科《法律咨询与调解》期末纸质考试试题及答案
- 护士条例解读
- 医务人员岗前培训课件
- 检修工(题库)附答案
- 2025届高考语文一轮复习:小说情节结构之伏笔 练习题(含答案)
- 《化学实验室安全》课程教学大纲
- 2024年人教版初二地理上册期末考试卷(附答案)
- 《一元一次方程》复习学案
- 三甲医院面试自我介绍课件
- 单招铁路基础知识题库单选题100道及答案解析
- 高低压供配电设备检查和检修保养合同3篇
评论
0/150
提交评论