日期:2014-05-16  浏览次数:20660 次

Oracle异常处理总结
为了处理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;
  

   
--3、处理自定义异常
预定义异常和非预定义异常都与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;
   
注意:不能在同一个block中描述EXCEPTION两次,但是可以描述一个exception在两个不同的block中。异常(exception)是有作用域的,子块的异常不能被当前块所捕捉,

 

--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|