学习资料:http://blog.csdn.net/lhy2199/article/details/7317688
? ? ? ? ? ? ? ? ?http://blog.csdn.net/vking_wang/article/details/9110899
1:前言
? ? ? ??任何ORACLE错误(报告为ORA-xxxxx形式的Oracle错误号)、PL/SQL运行错误或用户定义条件(不一写是错误),都可以处理。当然了,PL/SQL编译错误不能通过PL/SQL异常处理来处理,因为这些错误发生在PL/SQL程序执行之前。
? ? ? ? PL/SQL程序块一旦产生异常而没有指出如何处理时,程序就会自动终止整个程序运行。
2:分类
1. 预定义 ( Predefined )错误
? ? ? ? ORACLE预定义的异常情况大约有24个。对这种异常情况的处理,无需在程序中定义,由ORACLE自动将其引发。
2. 非预定义 ( Predefined )错误
? ? ? ? 即其他标准的ORACLE错误。对这种异常情况的处理,需要用户在程序中定义,然后由ORACLE自动将其引发。
3. 用户定义(User_define) 错误
? ? ? ? 程序执行过程中,出现编程人员认为的非正常情况。对这种异常情况的处理,需要用户在程序中定义,然后显式地在程序中将其引发。
?一般格式:
?
异常处理部分一般放在?PL/SQL?程序体的后半部,结构为:
EXCEPTION WHEN first_exception THEN <code to handle first exception > WHEN second_exception THEN <code to handle second exception > WHEN OTHERS THEN <code to handle others exception >END;?异常处理可以按任意次序排列,但?OTHERS?必须放在最后.
DECLARE 定义变量BEGIN sql语句EXCEPTION WHEN NO_DATA_FOUND THEN DBMS_OUTPUT.PUT_LINE('数据库中没有编码'); --见页尾异常表 WHEN OTHERS THEN DBMS_OUTPUT.PUT_LINE(SQLCODE||'---'||SQLERRM);END;?4:非预定义的异常处理(必须对非定义的ORACLE错误进行定义)
3)在PL/SQL?块的异常情况处理部分对异常情况做出相应的处理。
DECLARE deptno_remaining EXCEPTION; PRAGMA EXCEPTION_INIT(deptno_remaining, -2292); /* -2292 是违反一致性约束的错误代码 */BEGIN insert语句EXCEPTION WHEN deptno_remaining THEN DBMS_OUTPUT.PUT_LINE('违反数据完整性约束!'); WHEN OTHERS THEN DBMS_OUTPUT.PUT_LINE(SQLCODE||'---'||SQLERRM);END;?5:用户自定义的异常处理(用户定义的异常错误是通过显式使用?RAISE?语句来触发)
1) 在PL/SQL?块的定义部分定义异常情况:
3)在PL/SQL?块的异常情况处理部分对异常情况做出相应的处理。
DECLARE v_empno employees.employee_id%TYPE :=&empno; no_result EXCEPTION;BEGIN UPDATE employees SET salary = salary+100 WHERE employee_id = v_empno; IF SQL%NOTFOUND THEN --SQL%NOTFOUND表示sql语名没有影响数据库的数据 RAISE no_result; END IF;EXCEPTION WHEN no_result THEN DBMS_OUTPUT.PUT_LINE('你的数据更新语句失败了!'); WHEN OTHERS THEN DBMS_OUTPUT.PUT_LINE(SQLCODE||'---'||SQLERRM);END;?6:异常传播机智(异常出现的位置不同处理也不同)
? ? ? 如果当前块对该异常错误设置了处理,则执行它并成功完成该块的执行,然后控制转给包含块。
? ? ? 如果当前块的内部块发生错误,并没有处理,哪么外部块会进行处理
SQLERRM 返回遇到的Oracle错误信息.
CREATE TABLE errors (errnum NUMBER(4), errmsg VARCHAR2(100));DECLARE err_msg VARCHAR2(100);BEGIN /* 得到所有 ORACLE 错误信息 */ FOR err_num IN -100 .. 0 LOOP err_msg := SQLERRM(err_num); INSERT INTO errors VALUES(err_num, err_msg); END LOOP;END;DROP TABLE errors;?7:用户自定义异常处理:RAISE_APPLICATION_ERROR
语法如下:
???? RAISE_APPLICATION_ERROR(error_number, error_message,?[keep_errors]?);
说明
1)error_number?是从?–20,000?到?–20,999?之间的参数,这样就不会与 ORACLE 的任何错误代码发生冲突
2)error_message?是相应的提示信息(<?2048?字节)
3)keep_errors?为可选,如果keep_errors?=TRUE?,则新错误将被添加到已经引发的错误列表中。如果keep_errors=FALSE(缺省),则新错误将替换当前的错误列表。
错误号 | 异常错误信息名称 | 说明 |
ORA-0001 | Dup_val_on_index | 违反了唯一性限制 |
ORA-0051 | Timeout-on-resource | 在等待资源时发生超时 |
ORA-0061 | Transaction-backed-out | 由于发生死锁事务被撤消 |
ORA-1001 | Invalid-CURSOR | 试图使用一个无效的游标 |
ORA-1012 | Not-logged-on | 没有连接到ORACLE |
ORA-1017 | Login-denied | 无效的用户名/口令 |
ORA-1403 | No_data_found | SELECT INTO没有找到数据 |
ORA-1422 | Too_many_rows | SELECT INTO?返回多行 |
ORA-1476 | Zero-divide | 试图被零除 |
ORA-1722 | Invalid-NUMBER | 转换一个数字失败 |
ORA-6500 | Storage-error | 内存不够引发的内部错误 |
ORA-6501 | Program-error | 内部错误 |
ORA-6502 | Value-error | 转换或截断错误 |
ORA-6504 | Rowtype-mismatch | 宿主游标变量与?PL/SQL变量有不兼容行类型 |
ORA-6511 | CURSOR-already-OPEN | 试图打开一个已处于打开状态的游标 |
ORA-6530 | Access-INTO-null | 试图为null?对象的属性赋值 |
ORA-6531 | Collection-is-null | 试图将Exists?以外的集合( collection)方法应用于一个null pl/sql?表上或varray上 |
ORA-6532 | Subscript-outside-limit | 对嵌套或varray索引得引用超出声明范围以外 |
ORA-6533 | Subscript-beyond-count | 对嵌套或varray?索引得引用大于集合中元素的个数. ? |
?