| 
 异常的语法格式   
在begin语句内:   
exception   
when then   
when then   
when others then   
   
--异常处理  
--首先创建一份对象的用法  
create type xtype as object (name varchar2(20));  
   
declare  
 x xtype;  
begin  
 x.name:='aaa';  
exception  
 when ACCESS_INTO_NULL then  
 dbms_output.put_line('init x first');  
end;  
--不报错了 变成了一种正常的行为  
   
--case not found  
declare  
 x number(9);  
begin  
 x:=&x;  
 case  
 when x>100 then  
 dbms_output.put_line('>100');  
 when x>200 then  
 dbms_output.put_line('>200');  
 end case;  
exception  
 when CASE_NOT_FOUND then  
 dbms_output.put_line('case not found');  
end;  
oracle在预定义包STANDARD包中提供了一系列的预定义异常。他们是调试Oracle PL/SQL程序的有用工具。大部分错误用负号作为错误号。可以使用SQLCODE内置函数看到错误代码,使用SQLERRM得到异常的内置消息。   
 
  
   
    
    
    
   
   
    
    |   异常   |  
      错误   |  
      何时出现   |  
     
    
    |   ACCESS_INTO_NULL   |  
      ORA-06530   |  
      试图访问未初始化对象的时候出现   |  
     
    
    |   CASE_NOT_FOUND   |  
      ORA-06592   |  
      如果定义了一个没有ELSE子句的CASE语句,而且没有CASE语句满足运行时条件时出现该异常   |  
     
    
    |   COLLECTION_IS_NULL   |  
      ORA-06531   |  
      当程序去访问一个没有进行初始化的NESTED TABLE或者是VARRAY的时候,会出现该异常   |  
     
    
    |   CURSOR_ALREADY_OPEN   |  
      ORA-06511   |  
      游标已经被OPEN,如果再次尝试打开该游标的时候,会出现该异常   |  
     
    
    |   DUP_VAL_ON_INDEX   |  
      ORA-00001   |  
      如果插入一列被唯一索引约束的重复值的时候,就会引发该异常(该值被INDEX认定为冲突的)   |  
     
    
    |   INVALID_CURSOR   |  
      ORA-01001   |  
      不允许的游标操作,比如关闭一个已经被关闭的游标,就会引发   |  
     
    
    |   INVALID_NUMBER   |  
      ORA-01722   |  
      给数字值赋非数字值的时候,该异常就会发生,这个异常也会发生在批读取时候LIMIT子句返回非正数的时候   |  
     
    
    |   LOGIN_DENIED   |  
      ORA-01017   |  
      程序中,使用错误的用户名和密码登录的时候,就会抛出这个异常   |  
     
    
    |   NO_DATA_FOUND   |  
      ORA_06548   |  
      在使用SELECT INTO 结构,并且语句返回NULL值的时候;访问嵌套表中已经删除的表或者是访问INDEX BY表(联合数组)中的未初始化元素就会出现该异常   |  
     
    
    |   NOT_LOGGED_ON   |  
      ORA-01012   |  
      当程序发出数据库调用,但是没有连接的时候(通常,在实际与会话断开连接之后)   |  
     
    
    |   PROGRAM_ERROR   |  
      ORA-06501   |  
      当Oracle还未正式捕获的错误发生时常会发生,这是因为数据库大量的Object功能而发生   |  
     
    
    |   ROWTYPE_MISMATCH   |  
      ORA-06504   |  
      如果游标结构不适合PL/SQL游标变量或者是实际的游标参数不同于游标形参的时候发生该异常   |  
     
    
    |   SELF_IS_NULL   |  
      ORA-30625   |  
      调用一个对象类型非静态成员方法(其中没有初始化对象类型实例)的时候发生该异常   |  
     
    
    |   STORAGE_ERROR   |  
      ORA-06500   |  
      当内存不够分配SGA的足够配额或者是被破坏的时候,引发该异常   |  
     
    
    |   SUBSCRIPT_BEYOND_COUNT   |  
      ORA-06533   |  
      当分配给NESTED TABLE或者VARRAY的空间小于使用的下标的时候,发生该异常(类似于java的ArrayIndexOutOfBoundsException)   |  
     
    
    |   SUBSCRIPT_OUTSIDE_LIMIT   |  
      ORA-06532   |  
      使用非法的索引值来访问NESTED TABLE或者VARRAY的时候引发   |  
     
    
    |   SYS_INVALID_ROWID   |  
      ORA-01410   |  
      将无效的字符串转化为ROWID的时候引发   |  
     
    
    |   TIMEOUT_ON_RESOURCE   |  
      ORA-00051   |  
      当数据库不能安全锁定资源的时候引发   |  
     
    
    |   TOO_MANY_ROWS   |  
      ORA-01422   |  
      常见错误,在使用SELECT INTO 并且查询返回多个行时引发。如果子查询返回多行,而比较运算符为相等的时候也会引发该异常。   |  
     
    
    |   USERENV_COMMITSCN_ERROR   |  
      ORA-01725   |  
      只可使用函数USERENV('COMMITSCN')作为INSERT语句的VALUES子句中的顶级表达式或者作为UPDATE语句的SET子句中的右操作数   |  
     
    
    |   VALUE_ERROR   |  
      ORA-06502   |  
      将一个变量赋给另一个不能容纳该变量的变量时引发   |  
     
    
    |   ZERO_DIVIDE   |  
      ORA-01476   |  
      将某个数字除以0的时候,会发生该异常   |  
     
   
   
  
可以很方便的在SQL块中使用EXCEPTION来捕捉异常并且进行处理(当然,编译异常与在声明块中被抛出的异常除外,一个好的PL/SQL编程 习惯是规避在声明块中进行变量的赋值操作)。如果上面的预定义异常能够满足要求,就可以使用他们。如果不满足,则可以像JAVA程序一样,定义自己的异 常,并且可以使用RAISE EXCEPTION来抛出异常。   
 --pl/sql自定义的异常  比如更新数据时并没有找到数据没有数据进行改变 select * from myemp;
  declare        emp_nofound_exception exception; begin        update myemp set sal=sal+2 where empno=&empno;--如果输入的输一个字符串的话 就要用单引号括起来        if sql%notfound then          raise emp_nofound_exception;        else          dbms_output.put_line('ok');               end if; exception    when emp_nofound_exception then     dbms_output.put_line('no emp found!');   when others then      dbms_output.put_line('others'); end;
  --使用两个函数 --sqlcode  sqlerrm
  declare        emp_nofound_exception exception; begin        update myemp set sal=sal+2 where empno=&empno;--如果输入的输一个字符串的话 就要用单引号括起来        if sql%notfound then          raise emp_nofound_exception;        else          dbms_output.put_line('ok');               end if; exception    when others then      dbms_output.put_line('sqlcode='||sqlcode);     dbms_output.put_line('sqlerrm='||sqlerrm);--如果错误信息很长的话 可以用substr(sqlerrm,1,100)切割成一百的字符串 end;
 
 
  |