Oracle数据库之PL/SQL异常处理

Oracle数据库之PL/SQL异常处理

异常指的是在程序运行过程中发生的异常事件,通常是由硬件问题或者程序设计问题所导致的。

PL/SQL程序设计过程中,即使是写得最好的程序也可能会遇到错误或未预料到的事件。一个健壮的程序都应该能够正确处理各种异常情况,并尽可能从中恢复。

1. 异常处理

异常处理是用来处理正常执行过程中未预料的事件。PL/SQL程序块一旦产生异常而没有指出如何处理时,程序就会自动终止整个程序运行。

PL/SQL编程过程中,有三种类型的异常:

1.预定义异常

对这种异常情况的处理,无需在程序中定义,当PL/SQL程序违反Oracle规则或超越系统限制时隐式引发。

2.非预定义异常

其他标准的Oracle错误。对这种异常情况的处理,需要用户在程序中定义,然后由Oracle自动将其引发。

3.用户定义异常

程序执行过程中,出现编程人员认为的非正常情况。对这种异常情况的处理,需要用户在程序中定义,然后显式地在程序中将其引发。

异常处理通常放在PL/SQL程序的后部,语法结构为:

EXCEPTION
    WHEN { exception [ OR exception ]... | OTHERS }
        THEN statement [ statement ]...

2. 预定义的异常处理

常见预定义异常:

错误号 异常名称 说明
ORA-00001 DUP_VAL_ON_INDEX 重复索引值,违反了唯一性限制,当在唯一索引所对应的列上键入重复值时触发
ORA-01001 INVALID_CURSOR 试图使用一个无效的游标
ORA-01012 NOT_LOGGED_ON 没有连接到ORACLE
ORA-01017 LOGIN_DENIED 无效的用户名/口令
ORA-01403 NO_DATA_FOUND 没有找到数据时触发
ORA-01422 TOO_MANY_ROWS 返回多行
ORA-01722 INVALID_NUMBER 转换为数字失败时触发
ORA-06511 CURSOR_ALREADY_OPEN 试图打开一个已处于打开状态的游标
ORA-06592 CASE_NOT_FOUND 当case条件都不满足时触发

更多预定义异常见:http://docs.oracle.com/cd/E11882_01/appdev.112/e25519/errors.htm#LNPLS00703

对预定义异常的处理,只需在PL/SQL块的异常处理部分,直接引用相应的异常情况名,并对其完成相应的异常错误处理即可。

示例1:

DECLARE
  stock_price   NUMBER := 9.73;
  net_earnings  NUMBER := 0;
  pe_ratio      NUMBER;
BEGIN
  pe_ratio := stock_price / net_earnings;
  DBMS_OUTPUT.PUT_LINE(‘运算结果 = ‘ || pe_ratio);
EXCEPTION
  WHEN ZERO_DIVIDE THEN
    DBMS_OUTPUT.PUT_LINE(‘/ by zero‘);
    pe_ratio := NULL;
END;

运行结果:

/ by zero

为避免除0异常,可以采用如下示例2方式解决:

示例2:

DECLARE
  stock_price   NUMBER := 9.73;
  net_earnings  NUMBER := 0;
  pe_ratio      NUMBER;
BEGIN
  pe_ratio :=
    CASE net_earnings
      WHEN 0 THEN NULL
      ELSE stock_price / net_earnings
    END;
END;

示例3:

DECLARE
  default_number NUMBER := 0;
BEGIN
  INSERT INTO t VALUES(TO_NUMBER(‘100.00‘, ‘9G999‘));
EXCEPTION
  WHEN INVALID_NUMBER THEN
    DBMS_OUTPUT.PUT_LINE(‘使用默认值替换非法数字‘);
    INSERT INTO t VALUES(default_number);
END;

运行结果:

使用默认值替换非法数字

3. 非预定义的异常处理

非预定义异常有错误号没有名字,处理的办法是:自己定义一个名字,绑定到错误号,捕获错误名。处理这类异常,首先必须对非预定义的Oracle异常进行定义。

如:

myexcp EXCEPTION;

然后使用EXCEPTION_INIT语句与标准的ORACLE错误联系起来,如:

PRAGMA EXCEPTION_INIT(myexcp,-02292);

说明:ORA-02292是违反完整性约束的错误代码。

示例:

DECLARE
    myexcp EXCEPTION;
    PRAGMA EXCEPTION_INIT(myexcp,-02292);
    dno scott.emp.deptno%TYPE;
BEGIN
    dno := &dept_no;
    DELETE FROM scott.dept WHERE deptno=dno;
    EXCEPTION
        WHEN myexcp THEN
            DELETE FROM scott.emp WHERE deptno=dno;
            DELETE FROM scott.dept WHERE deptno=dno;
    END;

4. 用户定义异常处理

我们可以在任何PL/SQL匿名块,子程序或包的声明部分声明自己的异常。用户定义的异常是通过使用RAISE语句显式触发的。

一般用户定义异常的处理流程为:定义异常->抛出异常->捕获及处理异常。

示例:

DECLARE
   invalidCATEGORY EXCEPTION; -- 定义异常
   category VARCHAR2(10);
BEGIN
   category := ‘&Category‘;
   IF category NOT IN (‘附件‘,‘顶盖‘,‘备件‘) THEN
      RAISE invalidCATEGORY; -- 抛出异常
   ELSE
      DBMS_OUTPUT.PUT_LINE(‘您输入的类别是‘|| category);
   END IF;
EXCEPTION
   WHEN invalidCATEGORY THEN -- 捕获及处理异常
      DBMS_OUTPUT.PUT_LINE(‘无法识别该类别‘);
END;

我们可以调用RAISE_APPLICATION_ERROR过程引发并传播应用程序异常,这为应用程序提供了一种与ORACLE交互的方法。

RAISE_APPLICATION_ERROR过程可用于创建用户定义的错误信息,可以在可执行部分和异常处理部分使用,错误编号必须介于–20000 和–20999之间,错误消息的长度可长达2048个字节。

RAISE_APPLICATION_ERROR过程语法:

RAISE_APPLICATION_ERROR (error_code, message[, {TRUE | FALSE}]);

如果指定TRUE,PL/SQL把ERROR_CODE上的错误信息添加到堆栈的顶部。指定FALSE,PL/SQL替换ERROR_CODE错误堆栈,默认值为FALSE。

示例1:

DECLARE
   empno employees.employee_id%TYPE;
   no_such_row  EXCEPTION;
BEGIN
   empno := &empno;
   UPDATE employees SET salary = salary+100 WHERE id = empno;
   IF SQL%NOTFOUND THEN
      RAISE no_such_row;
   END IF;
EXCEPTION
   WHEN no_such_row THEN
      RAISE_APPLICATION_ERROR(-20001, ‘没有待修改的行‘);
END;

示例2:

BEGIN
   UPDATE emp SET deptno=80 WHERE empno=1111;
   IF SQL%NOTFOUND THEN
      RAISE_APPLICATION_ERROR(-20001,‘该雇员不存在!‘);
   END IF;
   EXCEPTION
     WHEN OTHERS THEN
        dbms_output.put_line(SQLCODE||‘-->‘||SQLERRM);
END;

SQLCODE用于取得Oracle错误号。

SQLERRM则用于取得与之相关的错误消息。

时间: 2024-12-21 22:00:11

Oracle数据库之PL/SQL异常处理的相关文章

Oracle数据库之PL/SQL过程与函数

Oracle数据库之PL/SQL过程与函数 PL/SQL块分为匿名块与命名块,命名块又包含子程序.包和触发器. 过程和函数统称为PL/SQL子程序,我们可以将商业逻辑.企业规则写成过程或函数保存到数据库中,以便共享. 过程和函数均存储在数据库中,并通过参数与其调用者交换信息.过程和函数的唯一区别是函数总向调用者返回数据,而过程不返回数据. 1. 存储过程概念 存储过程(Stored Procedure)是在大型数据库系统中,一组为了完成特定功能的SQL语句集,存储在数据库中.经过第一次编译后再次

Oracle数据库之PL/SQL程序设计简介

有利于客户/服务器环境应用的运行 对于客户/服务器环境来说,真正的瓶颈是网络上.无论网络多快,只要客户端与服务器进行大量的数据交换,应用运行的效率自然就回受到影响.如果使用PL/SQL进行编程,将这种具有大量数据处理的应用放在服务器执行,自然就省去了数据在网上的传输时间. 适合于客户环境 PL/SQL由于分为数据库PL/SQL部分和工具PL/SQL.对于客户端来说,PL/SQL可以嵌套到相应的工具中,客户端程序可以执行本地包含PL/SQL部分,也可以向服务发SQL命令或激活服务器端的PL/SQL

Oracle数据库之PL/SQL程序基础设计

http://www.cnblogs.com/hellokitty1/p/4572930.html DECLARE /* * 声明部分——定义常量.变量.复杂数据类型.游标.用户自定义异常 */ BEGIN /* * 执行部分——PL/SQL语句和SQL语句 */ EXCEPTION /* * 异常处理部分——处理运行异常 */ END; /*块结束标记 */ 要实现PL/SQL程序设计,先介绍如下的基本内容: 二.标识符 PL/SQL程序设计中的标识符定义与SQL的标识符定义的要求相同: 标识

Oracle数据库之PL/SQL程序设计基础

PL/SQL程序设计基础 一.PL/SQL块结构 前边我们已经介绍了PL/SQL块的结构,再来回顾一下: DECLARE /* * 声明部分——定义常量.变量.复杂数据类型.游标.用户自定义异常 */ BEGIN /* * 执行部分——PL/SQL语句和SQL语句 */ EXCEPTION /* * 异常处理部分——处理运行异常 */ END; /*块结束标记 */ 要实现PL/SQL程序设计,先介绍如下的基本内容: 二.标识符 PL/SQL程序设计中的标识符定义与SQL的标识符定义的要求相同:

ORACLE数据库之PL/SQL触发器、rownum、动态SQL、数据库之视图与索引

WHEN子句说明触发约束条件.Condition为一个逻辑表达时,其中必须包含相关名称,而不能包含查询语句,也不能调用PL/SQL函数.WHEN子句指定的触发约束条件只能用在BEFORE和AFTER行触发器中,不能用在INSTEAD OF行触发器和其它类型的触发器中. -- 创建记录操作事件的表 CREATE TABLE event_table( event VARCHAR2(50), time DATE ); -- 创建触发器 CREATE OR REPLACE TRIGGER tr_star

Oracle数据库之PL/SQL游标

1. 游标概念 字面意思是游动的光标,是指向上下文区域的句柄或指针. 在PL/SQL块中执行CRUD操作时,ORACLE会在内存中为其分配上下文区.用数据库语言来描述游标就是:映射在上下文区结果集中一行数据上的位置实体. 用户可以使用游标访问结果集中的任意一行数据,将游标指向某行后,即可对该行数据进行操作.游标为应用提供了一种对具有多行数据查询结果集中的每一行数据分别进行单独处理的方法,是设计嵌入式SQL语句的应用程序的常用编程方式. 在每个用户会话中,可以同时打开多个游标,其最大数量由数据库初

Oracle数据库之PL/SQL触发器

1. 介绍 触发器(trigger)是数据库提供给程序员和数据分析员来保证数据完整性的一种方法,它是与表事件相关的特殊的存储过程,它的执行不是由程序调用,也不是手工启动,而是由事件来触发,比如当对一个表进行操作(insert,delete,update)时就会激活它执行.触发器经常用于加强数据的完整性约束和业务规则等. ORACLE触发器有三种类型,分别是:DML触发器.替代触发器和系统触发器. DML触发器 顾名思义,DML触发器是由DML语句触发的.例如数据库的INSERT.UPDATE.D

oracle数据库使用PL/sql导入excel数据

1.打开PL/SQL工具,菜单进入   工具>ODBC导入器(菜单列表倒数第二): 2.连接数据库与选择导入的excel文件表: 3.选择导入的表以及excel与数据库表字段之间的对应关系: 原文地址:https://www.cnblogs.com/whitemouseV2-0/p/10837714.html

Oracle PL/SQL 异常处理

Oracle数据库中的异常:没有异常的转移,因为没有受检异常和非受检异常得区分. 1.异常的产生: 2.异常的处理: declare --变量定义,初始化赋值. begin --变量的赋值,函数调用,if,while等. exception --异常处理代码 when others then 异常处理语句. end: 3.异常的抛出:raise 4.多异常处理:Java的多异常是通过数据类型区分,Oracle数据库的多异常是通过异常编号区分. 区别不同的异常是实现多异常处理前提. declare