`
FengShen_Xia
  • 浏览: 273473 次
  • 性别: Icon_minigender_1
  • 来自: 东方水城
社区版块
存档分类
最新评论

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


oracle预定义的异常列表
2008-10-30 16:06
命名的系统异常         产生原因 
ACCESS_INTO_NULL         未定义对象 
CASE_NOT_FOUND         CASE 中若未包含相应的 WHEN ,并且没有设置 ELSE 时 
COLLECTION_IS_NULL         集合元素未初始化 
CURSER_ALREADY_OPEN         游标已经打开 
DUP_VAL_ON_INDEX         唯一索引对应的列上有重复的值 
INVALID_CURSOR         在不合法的游标上进行操作 
INVALID_NUMBER         内嵌的 SQL 语句不能将字符转换为数字 
NO_DATA_FOUND         使用 select into 未返回行,或应用索引表未初始化的元素时 
TOO_MANY_ROWS         执行 select into 时,结果集超过一行 
ZERO_DIVIDE         除数为 0 
SUBSCRIPT_BEYOND_COUNT         元素下标超过嵌套表或 VARRAY 的最大值 
SUBSCRIPT_OUTSIDE_LIMIT         使用嵌套表或 VARRAY 时,将下标指定为负数 
VALUE_ERROR         赋值时,变量长度不足以容纳实际数据 
LOGIN_DENIED         PL/SQL 应用程序连接到 oracle 数据库时,提供了不正确的用户名或密码 
NOT_LOGGED_ON         PL/SQL 应用程序在没有连接 oralce 数据库的情况下访问数据 
PROGRAM_ERROR         PL/SQL 内部问题,可能需要重装数据字典& pl./SQL 系统包 
ROWTYPE_MISMATCH         宿主游标变量与 PL/SQL 游标变量的返回类型不兼容 
SELF_IS_NULL         使用对象类型时,在 null 对象上调用对象方法 
STORAGE_ERROR         运行 PL/SQL 时,超出内存空间 
SYS_INVALID_ID         无效的 ROWID 字符串 
TIMEOUT_ON_RESOURCE         Oracle 在等待资源时超时

=============================================================

BEGIN
《PL/SQL块》;
Exception
when no_data_found then --没有找到数据
《响应命令》;
when too_many_rows then --返回多行,隐式光标每次只能检索一行数据
《响应命令》;
when invalid_number then --字符向数字转换失败
《响应命令》;
when zero_divide then --被零除
《响应命令》;
when dup_val_on_index then --向唯一索引中插入重复数据
《响应命令》;
when invalid_cursor then --非法游标操作
《响应命令》;
when value_error then --数字的,数据转换,截字符串或强制性的错误
《响应命令》;
when others then --发生其它任何错误
null; --选择一:什么也不做,就当错误没发生
raise form_trigger_failure; --选择二:挂起当前程序
END;


常用预定义例外
EXCEPTION
WHEN CURSOR_ALREADY_OPEN THEN -- ORA-06511 SQLCODE = -6511 游标已经打开
..WHEN DUP_VAL_ON_INDEX THEN -- ORA-00001 SQLCODE = -1 向唯一索引中插入重复数据
..WHEN INVALID_CURSOR THEN -- ORA-01001 SQLCODE = -1001 非法游标操作
..WHEN INVALID_NUMBER THEN -- ORA-01722 SQLCODE = -1722 字符向数字转换失败
..WHEN LOGIN_DENIED THEN -- ORA-01017 SQLCODE = -1017 

..WHEN NO_DATA_FOUND THEN -- ORA-01403 SQLCODE = +100 没有找到数据
..WHEN NOT_LOGGED_ON THEN -- ORA-01012 SQLCODE = -1012 
..WHEN PROGRAM_ERROR THEN -- ORA-06501 SQLCODE = -6501 程序错误
..WHEN STORAGE_ERROR THEN -- ORA-06500 SQLCODE = -6500
..WHEN TIMEOUT_ON_RESOURCE THEN -- ORA-00051 SQLCODE = -51

..WHEN TOO_MANY_ROWS THEN -- ORA-01422 SQLCODE = -1422 返回多行
..WHEN TRANSACTION_BACKED_OUT THEN -- ORA-00061 SQLCODE = -61

..WHEN VALUE_ERROR THEN -- ORA-06502 SQLCODE = -6502 数值转换错误
..WHEN ZERO_DIVIDE THEN -- ORA-01476 SQLCODE = -1476 被零除
..WHEN OTHERS THEN -- 其它任何错误的处理
..END; 

 

 

 

本文来自:http://blog.csdn.net/zjhiphop2006/archive/2010/06/01/5637578.aspx

 

 

分享到:
评论

相关推荐

    ORACLE 异常处理总结.doc

    Oracle 异常处理。即使是写得最好的PL/SQL程序也会遇到错误或未预料到的事件。一个优秀的程序都应该能够正确处理各种出错情况,并尽可能从错误中恢复。任何ORACLE错误(报告为ORA-xxxxx形式的Oracle错误号)、PL/SQL...

    自己总结的oracle常见异常处理步骤

    oracle异常处理步骤自己总结的,不太成熟,大家可以不看,只是我的备忘录

    oracle异常处理

    oracle异常处理 把各种异常和解决方式总结出来了

    oracle 转mysql项目总结

    (主要分事务处理,游标处理,存储过程方法调用,数组处理,异常处理等。) (2)oracle与mysql区别比较。 (主要包含:语法及结构区别,函数区别,数据类型区别等。) (3)ORACLE与MYSQL常用函数对比。

    Oracle数据库经典学习教程

    6. PL/SQL的异常处理 102 7. 本章总结 108 8. 本章练习 109 Oracle应用于.Net平台 111 1. 回顾ADO.NET 112 2. 使用ADO.NET连接Oracle 113 3. 抽象工厂中加入Oracle 117 4. 本章总结 121 5. 本章练习 122 数据库导入...

    oracle所有知识点笔记(全)

    流程控制,游标,异常处理,记录类型,视图, 控制用户权限,高级子查询,set运算符, 基本的sql_Select语句 运算符,多表联查,排序,组函数,序列,索引,同义词, 约束,创建和管理表,单行函数,过滤数据等等

    Oracle数据库学习指南

    15.PLSQL异常处理初步 16.SQL语句性能调整原则 17.创建和使用分区的表 18.基于成本的优化器一般错误概念和问题 19.Delphi 3_0中连接数据库的三种方式 20.远程数据库的访问 21.监控数据库性能的SQL 22...

    oracle实验报告

    (5)在未使用显式游标的情况下,使用SELECT语句必须保证只有一条记录返回,否则会产生异常情况。 [例3-1] 问题:编写一个过程,求和运算。 SET SERVEROUTPUT ON; DECLARE a number:=1; BEGIN a:=a+5; DBMS_...

    Oracle 10g应用指导

    介绍了PL/SQL中常用的函数、异常处理等,还有对随机数生成、分析函数、多表合并、多表插入等问题的解决方法。第7章 子程序和触发器,包括函数、存储过程、包以及触发器等。对子程序的调用者权限、管道表函数、传递...

    Oracle+10g应用指导与案例精讲

    介绍了PL/SQL中常用的函数、异常处理等,还有对随机数生成、分析函数、多表合并、多表插入等问题的解决方法。第7章 子程序和触发器,包括函数、存储过程、包以及触发器等。对子程序的调用者权限、管道表函数、传递...

    深入解析Oracle.DBA入门进阶与诊断案例

    针对数据库的启动和关闭、控制文件与数据库初始化、参数及参数文件、数据字典、内存管理、Buffer Cache与Shared Pool原理、重做、回滚与撤销、等待事件、性能诊断与SQL优化等几大Oracle热点主题,...10.7 总结 525

    2021 云和恩墨大讲堂PPT汇总(50份).zip

    深入解析:oracle drop table purge内部原理及异常恢复 数据时代万象更新-从数据库技术演进看国产数据库机遇 一次特殊的Oralce 硬解析性能问题的技术分享 CloudQuery 权限管理体系在证券行业的应用 DML操作时索引是...

    Oracl技术资料(EBook)

    15.PLSQL异常处理初步 16.SQL语句性能调整原则 17.创建和使用分区的表 18.基于成本的优化器一般错误概念和问题 19.Delphi 3_0中连接数据库的三种方式 20.远程数据库的访问 21.监控数据库性能的SQL 22.简单...

    ORACLE速成手册 面向应用

    6. PL/SQL的异常处理 ...................................................................................................... 109 7. 本章总结 ................................................................

    PL/SQL 详解

    ORACLE 学习文档,总结,囊括常用过程,函数,游标,异常处理。。。,非常实用

    世博安保系统-性能测试报告

    世博安保系统-性能测试报告: 1.1 性能测试目标 1.2 性能测试总体结果 1.2.1 性能评估等级分布图 1.2.2 性能测试结果描述: 2 系统测试摘要 ... 3.4.4 Oracle 数据库异常 4 总结 5 附:世博安保系统截图

    JAVA(J2SEJ2EE)学习笔记pdf

     内容包括:j2se语法、循环控制、OOP设计、方法设计、重写、重载、抽象类/方法、接口、异常处理、多线程、swing。以及部分oracle内容。   本笔记是pdf格式,经过我多次排版,结构清晰接近出版物,内容充实,示例...

    SQL21日自学通

    异常的处理391 将输入返回给用户392 在PL/SQL 中的事务控制393 让所有的事在一起工作394 示例表及数据394 一个简单的PL/SQL 语句块395 又一个程序398 存储过程包和触发机制403 总结406 问与答407 校练场407 练习407 ...

    java 面试题 总结

    finally是异常处理语句结构的一部分,表示总是执行。 finalize是Object类的一个方法,在垃圾收集器执行的时候会调用被回收对象的此方法,可以覆盖此方法提供垃圾收集时的其他资源回收,例如关闭文件等。 13、sleep()...

    整理后java开发全套达内学习笔记(含练习)

    //此语句会抛异常,需处理 System.out.println("您输入了文字:" + next); }catch(Exception e){} }} 数值保存方式: 正数= 二进制 负数= 补码 补码= 反码 +1 正数=负数的补码(反码+1) 反码= 非(二...

Global site tag (gtag.js) - Google Analytics