分享我对ORACLE 异常处理的总结,
ERP100文章发表后,会去掉着色,
有语法着色的,大家可以浏览这儿: http://www.ouora.com/post/145.html
为了处理 PL/SQL 应用程序的各种错误,开发人员可以使用各种类型的异常。
Oracle 提供了
1:预定义异常
用于处理常见的 Oracle 错误
2:非预定义异常
用于处理预定义异常所不能处理的 Oracle 错误
3:自定义异常
用于处理于 Oracle 错误无关的其他情况
异常处理部分是以关键字 EXCEPTION 开始的,语法如下:
异常处理部分从关键字 EXCEPTION 开始,在异常处理部分使用 WHEN 字句捕捉各种异常,如果有其他未预定义到的异常,使用 WHEN OTHERS THEN 字句进行捕捉和处理。
1 、 处理预定义异常,这是系统预定的 21种类型
2 、处理非预定义异常
使用非预定义异常包括三步:
一:在定义部分定义异常名,
二:在异常和 Oracle 错误之间建立关联,
三:在异常处理部分捕捉并处理异常。
当定义 Oracle 错误和异常之间的关联关系时,需要使用伪过程 EXCEPTION_INIT 。
一:首先的定义部分定义异常;
二:使用 progmaexception_init(exception_name,exception_number) 在异常和 oracle 错误之间建立关联,
这时要求用户知道可能出现的错误号(异常函数 sqlcode 、 sqlerrm 和 raise_application_error );
三:最终在异常处理部分捕捉并处理异常。
下面以更新特定雇员的部门号,并处理 ORA-02291 错误为例,说明使用非预定义异常的方法。示例如下:
--3
、处理自定义异常
预定义异常和非预定义异常都与 Oracle 错误有关,并且当出现 Oracle 错误时会隐含触发相应异常;
而自定义异常与 Oracle 错误没有任何关联,它是由开发人员为特定情况所定义的异常。
当使用自定义异常时,
一:需要在定义部分( DECLARE )定义异常,
二:再执行部分( BEGIN )触发异常(使用 RAISE 语句),
三:在异常处理部分( EXCEPTION )捕捉并处理异常。
注意:不能在同一个
block
中描述
EXCEPTION
两次,但是可以描述一个
exception
在两个不同的
block
中。异常(
exception
)是有作用域的,子块的异常不能被当前块所捕捉,
--4、使用异常函数:
Oracle 内置函数 sqlcode 和 sqlerrm 主要用在 others 处理器中,分别用来返回 oracle 的错误代码和错误消息。
一般情况下 sqlcode 返回负数标识的 oracle 错误代码,除非错误
为 ‘ora-01403 : no data found’ ,此时对应的 sqlcode 为 +100 ,
对于用户自定义的异常, sqlcode 返回 +1 ,如果没有异常被触发, sqlcode 返回 0 。
Oracle 过程 raise_application_error 用于在 plsql 应用程序中自定义错误消息。
注意该过程只能在数据库端的子程序(过程、函数、包、触发器)中使用,而不能在匿名块和客户端的子程序中使用。
语法为 raise_application_error(error_number,message[,[true|false]]);
其中
error_number 用于定义错误号,该错误号必须在 -20000 到 -20999 之间的负整数;
message 用于指定错误消息,并且该消息的长度不
能超过 2048 字节;
第三个参数如果为 true ,则该错误会被放在先前错误堆栈中,如果为 false (默认值)则会替代先前所有错误。
-- 5、 plsql 编译警告:
plsql 警告可以分为四类:
severe :用于
检查可能出现的不可预料或者错误结果,例如参数的别名问题;
performance :用于检查可能引起的性能问题,例如执行 insert 操作时为 number 列提供了 varchar2 类型数据;
informational :用于检查子程序中的死代码;
all :用于检查所有警告。
为了数据库可以在编
译 plsql 子程序时发出警告信息,需要设置初始化参数 plsql_warnings 。这个参数不仅可以在系统级或者会话级设置,也可以在 alter procedure 命令中设置。
为了检查是否存在对应警告信息,必须先激活警告检查,然后重新编译子程序,
最后使用 show errors 命令显示警告错误。
--6、定义Exception时要注意的一些事项
当异常发生时,在块的内部没有该异常处理器时,控制将转到或传播到上一层块的异常处理部分。
没有处理的异常将沿检测异常调用程序传播到外层,当异常被处理并解决或到达程序最外层传播停止。在声明部分抛出的异常将控制转到上一层的异常部分。
用户必须在独立的 WHEN 子串中为每个异常设计异常处理代码, WHEN OTHERS 子串必须放置在最后面作为缺省处理器处理没有显式处理的异常。当异常发生时,控制转到异常部分, ORACLE 查找当前异常相应的 WHEN..THEN 语句,捕捉异常, THEN 之后的代码被执行,如果错误陷阱代码只是退出相应的嵌套块,那么程序将继续执行内部块 END 后面的语句。如果没有找到相应的异常陷阱,那么将执行 WHEN OTHERS 。在异常部分 WHEN 子串没有数量限制。
ERP100文章发表后,会去掉着色,
有语法着色的,大家可以浏览这儿: http://www.ouora.com/post/145.html
为了处理 PL/SQL 应用程序的各种错误,开发人员可以使用各种类型的异常。
Oracle 提供了
1:预定义异常
用于处理常见的 Oracle 错误
2:非预定义异常
用于处理预定义异常所不能处理的 Oracle 错误
3:自定义异常
用于处理于 Oracle 错误无关的其他情况
异常处理部分是以关键字 EXCEPTION 开始的,语法如下:
EXCEPTION WHEN exception_Name THEN --exception_Name为异常的名字 statement1; WHEN OTHERS THEN statement1; |
异常处理部分从关键字 EXCEPTION 开始,在异常处理部分使用 WHEN 字句捕捉各种异常,如果有其他未预定义到的异常,使用 WHEN OTHERS THEN 字句进行捕捉和处理。
1 、 处理预定义异常,这是系统预定的 21种类型
错误代码是负整数,如-51 | |
Access_info_null(ora-06530) | 当访问没有初始化的对象时触发。 |
Case_not_found(ora-06592) | case过程中when后没有包含必要的条件分支并且没有else子句,则会触发本异常。 |
Collection_is_null(06531) | 访问未初始化的集合元素(嵌套表或者varray)。 |
Cursor_already_open(ora-06511) | 重新打开已经打开的游标。 |
Dup_val_on_index(ora-00001) | 当中唯一索引所对应的列上键入重复值时。 |
Invalid_cursor(ora-01001) | 试图在不合法的游标上执行操作时,譬如没打开游标就提取内容 |
Invalid_number(ora-01722) | 当试图将非法的字符串转换为数字类型时。 |
No_data_found(ora-01403) | 执行select into未返回行,或者引用了索引表未初始化的元素时。 |
Too_many_rows(ora-01422) | 执行select into返回超过一行数据时。 |
Zero_divide(ora-01476) | 0作为被除数时。 |
Subscript_beyond_count(ora-06533) | 使用嵌套表或者varray集合时,如果引用下标超过last。 |
Subscript_outside_limit(ora-06532) | 使用嵌套表或varray集合时,如果引用下标小于first。 |
Value_error(ora-06502) | 在执行赋值操作时,如果变量长度不足以容纳实际数据。 |
Login_denied(ora-01017) | 连接数据库时提供了不正确的用户名或口令。 |
Not_logged_on(ora-01012) | 在程序没有连接到oracle数据库时执行plsql代码则会触发。 |
Program_error(ora-06501) | plsql内部问题。 |
Rowtype_mismatch(ora-06504) | 执行赋值操作时,如果宿主游标变量和PLSQL游标变量返回类型不兼容时。 |
Self_is_null(ora-30625) | 使用对象类型时,如果在null实例上调用成员方法。 |
Storage_error(ora-06500) | 超出内存空间或者内存被损坏。 |
Sys_invalid_rowid(ora-01410) | 无效字符串企图转换为rowid类型时。 |
Timeout_on_resource(ora-00051) | 等待资源时出现超时错误。 |
2 、处理非预定义异常
使用非预定义异常包括三步:
一:在定义部分定义异常名,
二:在异常和 Oracle 错误之间建立关联,
三:在异常处理部分捕捉并处理异常。
当定义 Oracle 错误和异常之间的关联关系时,需要使用伪过程 EXCEPTION_INIT 。
一:首先的定义部分定义异常;
二:使用 progmaexception_init(exception_name,exception_number) 在异常和 oracle 错误之间建立关联,
这时要求用户知道可能出现的错误号(异常函数 sqlcode 、 sqlerrm 和 raise_application_error );
三:最终在异常处理部分捕捉并处理异常。
下面以更新特定雇员的部门号,并处理 ORA-02291 错误为例,说明使用非预定义异常的方法。示例如下:
DECLARE e_integrity EXCEPTION; --1、定义部分 PRAGMA EXCEPTION_INIT (e_integrity, -2291); --2、建立关联关系 BEGIN UPDATE emp SET deptno= &dno WHERE empno = &eno; EXCEPTION WHEN e_integrity THEN --3、捕捉处理 DBMS_OUTPUT.PUT_LINE(‘该部门不存在’); END; |
预定义异常和非预定义异常都与 Oracle 错误有关,并且当出现 Oracle 错误时会隐含触发相应异常;
而自定义异常与 Oracle 错误没有任何关联,它是由开发人员为特定情况所定义的异常。
当使用自定义异常时,
一:需要在定义部分( DECLARE )定义异常,
二:再执行部分( BEGIN )触发异常(使用 RAISE 语句),
三:在异常处理部分( EXCEPTION )捕捉并处理异常。
declare myexception exception; begin if 1=0 then raise myexception; endif; exception when myexception then dbms_output.put_line('asdf'); end; |
--4、使用异常函数:
Oracle 内置函数 sqlcode 和 sqlerrm 主要用在 others 处理器中,分别用来返回 oracle 的错误代码和错误消息。
一般情况下 sqlcode 返回负数标识的 oracle 错误代码,除非错误
为 ‘ora-01403 : no data found’ ,此时对应的 sqlcode 为 +100 ,
对于用户自定义的异常, sqlcode 返回 +1 ,如果没有异常被触发, sqlcode 返回 0 。
Begin Exception When others then Dbms_output.put_line(sqlcode||sqlerrm(sqlcode)); End; |
Oracle 过程 raise_application_error 用于在 plsql 应用程序中自定义错误消息。
注意该过程只能在数据库端的子程序(过程、函数、包、触发器)中使用,而不能在匿名块和客户端的子程序中使用。
语法为 raise_application_error(error_number,message[,[true|false]]);
其中
error_number 用于定义错误号,该错误号必须在 -20000 到 -20999 之间的负整数;
message 用于指定错误消息,并且该消息的长度不
能超过 2048 字节;
第三个参数如果为 true ,则该错误会被放在先前错误堆栈中,如果为 false (默认值)则会替代先前所有错误。
IF product_not_found THEN RAISE_APPLICATION_ERROR(-20123,'Invald product code' TRUE); END IF; |
-- 5、 plsql 编译警告:
plsql 警告可以分为四类:
severe :用于
检查可能出现的不可预料或者错误结果,例如参数的别名问题;
performance :用于检查可能引起的性能问题,例如执行 insert 操作时为 number 列提供了 varchar2 类型数据;
informational :用于检查子程序中的死代码;
all :用于检查所有警告。
为了数据库可以在编
译 plsql 子程序时发出警告信息,需要设置初始化参数 plsql_warnings 。这个参数不仅可以在系统级或者会话级设置,也可以在 alter procedure 命令中设置。
Alter {system|session|procedure} set plsql_warnings= ’{enable|disable:{all |performance|severe|informational}}’; |
为了检查是否存在对应警告信息,必须先激活警告检查,然后重新编译子程序,
最后使用 show errors 命令显示警告错误。
create or replace procedure my_test is begin if 1=0 then dbms_output.put_line('test'); endif; end; SQL> alter procedure my_test compile plsql_warnings = 'enable:all'; Procedure altered SQL> show errors; Errors for PROCEDURE SYS.MY_TEST: LINE/COL ERROR -------- ------------------------- 10/5 PLW-06002: 无法执行的代码 |
--6、定义Exception时要注意的一些事项
当异常发生时,在块的内部没有该异常处理器时,控制将转到或传播到上一层块的异常处理部分。
没有处理的异常将沿检测异常调用程序传播到外层,当异常被处理并解决或到达程序最外层传播停止。在声明部分抛出的异常将控制转到上一层的异常部分。
用户必须在独立的 WHEN 子串中为每个异常设计异常处理代码, WHEN OTHERS 子串必须放置在最后面作为缺省处理器处理没有显式处理的异常。当异常发生时,控制转到异常部分, ORACLE 查找当前异常相应的 WHEN..THEN 语句,捕捉异常, THEN 之后的代码被执行,如果错误陷阱代码只是退出相应的嵌套块,那么程序将继续执行内部块 END 后面的语句。如果没有找到相应的异常陷阱,那么将执行 WHEN OTHERS 。在异常部分 WHEN 子串没有数量限制。
EXCEPTION WHEN inventory_too_low THEN ...... WHEN discontinued_item THEN ...... WHEN zero_divide THEN ...... WHEN OTHERS THEN ...... |
来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/24850250/viewspace-688651/,如需转载,请注明出处,否则将追究法律责任。
转载于:http://blog.itpub.net/24850250/viewspace-688651/