Oracle PL/SQL中异常高级特性示例解析
PL/SQL(ProceduralLanguage/SQL,过程语言/SQL)是结合了Oracel过程语言和结构化查询语言(SQL)的一种扩展语言。
优点:
(1)PL/SQL具有编程语言的特点,它能把一组SQL语句放到一个模块中,使其更具模块化种序的特点。
(2)PL/SQL可以采用过程性语言控制程序的结构。
(3)PL/SQL有自动处理的异常处理机制。
(4)PL/SQL程序块具有更好的可移植性,可移植到另一个Oracle数据库中。
(5)PL/SQL程序减少了网络的交互,有助于提高程序性能。
在OraclePL/SQL语句块中exception的异常处理部分是非常重要的组成部分,它决定了在PL/SQL语句块内部可执行部分在发生异常错误时,程序是友好地提示:程序遇到某些错误而无法执行,还是抛出一堆难以理解的Oracle内部错误码。
本文只介绍3种PL/SQL异常的三种高级形态,用于解决Oracle内置异常过少,很多时候不能够满足实际的使用需求。
1,RAISE_APPLICATION_ERROR
-是Oracle提供的一种特殊的内置过程,允许程序员为特定的程序创建有意义的错误消息,适用于用户自定义定义异常。
-语法结构
RAISE_APPLICATION_ERROR(error_number,error_message);或者
RAISE_APPLICATION_ERROR(error_number,error_message,keep_errors)
-error_number是与特定错误消息关联的错误编号,Oracle预留了-20999---20000专门提供给程序员自定义错误代码。
-error_message是错误消息文本,最多包含2048个字符。
-keep_errors是可选的Boolean参数,默认为FALSE,如果为TRUE,新抛出的错误会被添加到已抛出的错误列表中,这个错误列表称为错误栈,如果为FALSE,新错误会替换已抛出的错误栈。
-适用于未命名的用户定义异常,负责把错误编号和错误消息关联,用户定义了异常,却没有定义该错误的名称
-使用RAISE_APPLICATION_ERROR过程,程序员能够遵循与Oracle一致的方式返回错误消息。
-示例代码
declare v_idnumber:=&p_id; v_namevarchar2(20); v_salnumber; begin ifv_id>0then selectename,salintov_name,v_salfromempwhereempno=v_id; dbms_output.put_line(chr(10)||v_name||''||v_sal); else raise_application_error(-20001,'Employeeidcannotbenegative.'); endif; exception whenNO_DATA_FOUNDthen dbms_output.put_line(chr(10)||'Thereisnosuchemployeeidis'||v_id); end; / Entervalueforp_id:40 old2:v_idnumber:=&p_id; new2:v_idnumber:=40; Thereisnosuchemployeeidis40 PL/SQLproceduresuccessfullycompleted. / Entervalueforp_id:-90 old2:v_idnumber:=&p_id; new2:v_idnumber:=-90; declare * ERRORatline1: ORA-20001:Employeeidcannotbenegative. ORA-06512:atline11
-示例解析:该PL/SQL代码会根据用户输入的员工Id,查询员工的姓名和工资。当我们输入存在的员工编号时,程序能够正常返回结果;如果输入不存在ID,则selectinto语句会抛出没有返回行,进而使程序进入异常处理部分(本部分为举例),程序同样执行成功;当输入一个负数时,if条件语句就会进入到raise_application_error部分,由于可执行部分运行发生错误,执行焦点会立即转移到异常处理部分,而异常处理部分没有关于该异常的处理,所以程序报错,并返回到用户界面。
-是哟个raise_application_error,程序员可以使程序实现像Oracle系统产生的错误消息。
-事实上,单纯使用raise_application_error,因为没有异常的名称,如果要对其进行异常处理,只能够使用others(下文有专门的介绍)。
2,EXCEPTION_INIT
-使用EXCEPTION_INIT编译指令,可以将用户自定义的Oracle错误编号和用户自定义的错误名称关联起来,相当于用户自定义错误和RAISE_APPLICATION_ERROR的结合体。
-EXCEPTION_INIT出现在语句块的声明部分:
exception_nameexception; pragmaexception_init(exception_name,error_code)
-考虑如下代码:
declare v_nonumber:=&p_no; begin deletefromdeptwheredeptno=v_no; dbms_output.put_line(chr(10)||'Thedepartmentidis'||v_no||'hasbeendeleted'); end; / Entervalueforp_no:20 old2:v_nonumber:=&p_no; new2:v_nonumber:=20; declare * ERRORatline1: ORA-02292:integrityconstraint(SCOTT.FK_DEPTNO)violated-childrecordfound ORA-06512:atline4
-由于违反外键约束,删除部门失败了。但是抛出的错误不是很好理解
-我们可以使用EXCEPTION_INIT来对这个错误进行处理,首先我们得知道违反外键约束的这个Oracle错误代码“ORA-02292”
-使用EXCEPTION_INIT
declare v_nonumber:=&p_no; e_dept_existexception; pragmaexception_init(e_dept_exist,-02292); begin deletefromdeptwheredeptno=v_no; dbms_output.put_line(chr(10)||'Thedepartmentidis'||v_no||'hasbeendeleted'); exception whene_dept_existthen dbms_output.put_line(chr(10)||'Therearesomeemployeesinthisdeptartment,ifyouwantdeletethisdeptartment,pleasedeletetheseemployeesinthedepartmentfirst.'); end; / Entervalueforp_no:20 old2:v_nonumber:=&p_no; new2:v_nonumber:=20; Therearesomeemployeesinthisdeptartment,ifyouwantdeletethisdeptartment,pleasedeletetheseemployeesinthedepartmentfirst. PL/SQLproceduresuccessfullycompleted.
-这下抛出的错误就容易理解多了。首先我们定义了一个名为e_dept_exist的异常,然后将这个异常与Oracle错误代码-02292进行关联。当程序执行报错时进入异常处理部分,在这里我们重新给这个错误定义了错误消息。
3,SQLCODE和SQLERRM
-在异常处理中,当异常的名称未知时(比如上面1中RAISE_APPLICATION_ERROR),都可以使用others来进行异常的捕获处理;
-由于others所捕获的异常是未知的(也可以是已知的,但是在程序中没有将其枚举出来),因此需要使用Oracle提供的两个内置函数SQLCODE、SQLERRM来针对others的异常进行处理:
-SQLCODE会返回Oracle的错误编号
-SQLERRM,返回错误的消息
-示例1,处理Oracle系统返回的错误:
declare v_nonumber:=&p_no; error_codenumber; error_msgvarchar2(500); begin deletefromdeptwheredeptno=v_no; dbms_output.put_line(chr(10)||'Thedepartmentidis'||v_no||'hasbeendeleted'); exception whenothersthen error_code:=sqlcode; error_msg:=sqlerrm; dbms_output.put_line(chr(10)||'Errorcodeis:'||error_code); dbms_output.put_line(chr(10)||'Errormessageis:'||error_msg); end; Entervalueforp_no:10 old2:v_nonumber:=&p_no; new2:v_nonumber:=10; Errorcodeis:-2292 Errormessageis:ORA-02292:integrityconstraint(SCOTT.FK_DEPTNO)violated-childrecordfound PL/SQLproceduresuccessfullycompleted.
-请注意exception异常处理部分,在该部分里面我们用到了声明部分定义的两个变量,error_code用来存储SQLCODE,error_msg用来存储SQLERRM。然后将两个变量值打印出来。
-示例2,处理用户自定义的异常:
declare v_idnumber:=&p_id; v_namevarchar2(20); v_salnumber; begin ifv_id>0then selectename,salintov_name,v_salfromempwhereempno=v_id; dbms_output.put_line(chr(10)||v_name||''||v_sal); else raise_application_error(-20001,'Employeeidcannotbenegative.'); endif; exception whenNO_DATA_FOUNDthen dbms_output.put_line(chr(10)||'Thereisnosuchemployeeidis'||v_id); whenothersthen declare error_codenumber; error_msgvarchar2(500); begin error_code:=sqlcode; error_msg:=sqlerrm; dbms_output.put_line(chr(10)||'Errorcodeis:'||error_code); dbms_output.put_line(chr(10)||'Errormessageis:'||error_msg); end; end; / Entervalueforp_id:-90 old2:v_idnumber:=&p_id; new2:v_idnumber:=-90; Errorcodeis:-20001 Errormessageis:ORA-20001:Employeeidcannotbenegative. PL/SQLproceduresuccessfullycompleted.
-在本代码中使用了raise_application_error,由于单纯的使用raise_application_error,只能使用others进行捕获。在异常处理部分,我们使用了一个PL/SQL语句块来处理这个错误,声明两个变量,并将SQLCODE和SQLERRM以字面值赋值的方法给这两个变量。
总结
以上所述是小编给大家介绍的OraclePL/SQL中异常高级特性示例解析,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对毛票票网站的支持!