即便是寫得最好的PL/SQL程序也會遇到錯誤或未預料到的事件。一個優秀的程序都應該可以正確處理各類出錯狀況,並儘量從錯誤中恢復。任何ORACLE錯誤(報告爲ORA-xxxxx形式的Oracle錯誤號)、PL/SQL運行錯誤或用戶定義條件(不一寫是錯誤),均可以。固然了,PL/SQL編譯錯誤不能經過PL/SQL異常處理來處理,由於這些錯誤發生在PL/SQL程序執行以前。
ORACLE 提供異常狀況(EXCEPTION)和異常處理(EXCEPTION HANDLER)來實現錯誤處理。sql
1 異常處理概念
異常狀況處理(EXCEPTION)是用來處理正常執行過程當中未預料的事件,程序塊的異常處理預約義的錯誤和自定義錯誤,因爲PL/SQL程序塊一旦產生異常而沒有指出如何處理時,程序就會自動終止整個程序運行.數據庫
有三種類型的異常錯誤:
1. 預約義 ( Predefined )錯誤
ORACLE預約義的異常狀況大約有24個。對這種異常狀況的處理,無需在程序中定義,由ORACLE自動將其引起。
2. 非預約義 ( Predefined )錯誤
即其餘標準的ORACLE錯誤。對這種異常狀況的處理,須要用戶在程序中定義,而後由ORACLE自動將其引起。
3. 用戶定義(User_define) 錯誤
程序執行過程當中,出現編程人員認爲的非正常狀況。對這種異常狀況的處理,須要用戶在程序中定義,而後顯式地在程序中將其引起。編程
異常處理部分通常放在 PL/SQL 程序體的後半部,結構爲:ide
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 必須放在最後.3d
1 預約義的異常處理
預約義說明的部分 ORACLE 異常錯誤
全部錯誤代碼和描述,文件在$ORACLE_HOME/rdbms/mesg/oraus.msg
對這種異常狀況的處理,只需在PL/SQL塊的異常處理部分,直接引用相應的異常狀況名,並對其完成相應的異常錯誤處理便可。
例1:更新指定員工工資,如工資小於1500,則加100;
D:\instantclient_12_2>sqlplus hr/hr@pdbtestcode
SQL*Plus: Release 12.2.0.1.0 Production on Thu Dec 21 09:55:15 2017orm
Copyright (c) 1982, 2016, Oracle. All rights reserved.server
Last Successful login time: Wed Dec 20 2017 10:59:38 +08:00
Connected to:
Oracle Database 12c Enterprise Edition Release 12.2.0.1.0 - 64bit Production
SQL> set serveroutput on
SQL> DECLARE
2 v_empno employees.employee_id%TYPE := &empno;
3 v_sal employees.salary%TYPE;
4 BEGIN
5 SELECT salary INTO v_sal FROM employees WHERE employee_id = v_empno;
6 IF v_sal<=1500 THEN
7 UPDATE employees SET salary = salary + 100 WHERE employee_id=v_empno;
8 DBMS_OUTPUT.PUT_LINE('編碼爲'||v_empno||'員工工資已更新!');
9 ELSE
10 DBMS_OUTPUT.PUT_LINE('編碼爲'||v_empno||'員工工資已經超過規定值!');
11 END IF;
12 EXCEPTION
13 WHEN NO_DATA_FOUND THEN
14 DBMS_OUTPUT.PUT_LINE('數據庫中沒有編碼爲'||v_empno||'的員工');
15 WHEN TOO_MANY_ROWS THEN
16 DBMS_OUTPUT.PUT_LINE('程序運行錯誤!請使用遊標');
17 WHEN OTHERS THEN
18 DBMS_OUTPUT.PUT_LINE(SQLCODE||'---'||SQLERRM);
19 END;
20 /
Enter value for empno: 7782
old 2: v_empno employees.employee_id%TYPE := &empno;
new 2: v_empno employees.employee_id%TYPE := 7782;
數據庫中沒有編碼爲7782的員工
PL/SQL procedure successfully completed.
2 非預約義的異常處理
對於這類異常狀況的處理,首先必須對非定義的ORACLE錯誤進行定義。步驟以下:
<異常狀況> EXCEPTION;
將其定義好的異常狀況,與標準的ORACLE錯誤聯繫起來,使用EXCEPTION_INIT語句:
PRAGMA EXCEPTION_INIT(<異常狀況>, <錯誤代碼>);
例1:刪除指定部門的記錄信息,以確保該部門沒有員工。
SQL> DECLARE
2 v_deptno departments.department_id%TYPE := &deptno;
3 deptno_remaining EXCEPTION;
4 PRAGMA EXCEPTION_INIT(deptno_remaining, -2292);
5 / -2292 是違反一致性約束的錯誤代碼 /
6 BEGIN
7 DELETE FROM departments WHERE department_id = v_deptno;
8 EXCEPTION
9 WHEN deptno_remaining THEN
10 DBMS_OUTPUT.PUT_LINE('違反數據完整性約束!');
11 WHEN OTHERS THEN
12 DBMS_OUTPUT.PUT_LINE(SQLCODE||'---'||SQLERRM);
13 END;
14 /
Enter value for deptno: 50
old 2: v_deptno departments.department_id%TYPE := &deptno;
new 2: v_deptno departments.department_id%TYPE := 50;
違反數據完整性約束!
PL/SQL procedure successfully completed.
3 用戶自定義的異常處理
當與一個異常錯誤相關的錯誤出現時,就會隱含觸發該異常錯誤。用戶定義的異常錯誤是經過顯式使用 RAISE 語句來觸發。當引起一個異常錯誤時,控制就轉向到 EXCEPTION塊異常錯誤部分,執行錯誤處理代碼。
對於這類異常狀況的處理,步驟以下:
1. 在PL/SQL 塊的定義部分定義異常狀況:
<異常狀況> EXCEPTION;
2. RAISE <異常狀況>;
3. 在PL/SQL 塊的異常狀況處理部分對異常狀況作出相應的處理。
例1:更新指定員工工資,增長100;
SQL> DECLARE
2 v_empno employees.employee_id%TYPE :=&empno;
3 no_result EXCEPTION;
4 BEGIN
5 UPDATE employees SET salary = salary+100 WHERE employee_id = v_empno;
6 IF SQL%NOTFOUND THEN
7 RAISE no_result;
8 END IF;
9 EXCEPTION
10 WHEN no_result THEN
11 DBMS_OUTPUT.PUT_LINE('你的數據更新語句失敗了!');
12 WHEN OTHERS THEN
13 DBMS_OUTPUT.PUT_LINE(SQLCODE||'---'||SQLERRM);
14 END;
15 /
Enter value for empno: 5000
old 2: v_empno employees.employee_id%TYPE :=&empno;
new 2: v_empno employees.employee_id%TYPE :=5000;
你的數據更新語句失敗了!
PL/SQL procedure successfully completed.
4 用戶定義的異常處理
調用DBMS_STANDARD(ORACLE提供的包)包所定義的RAISE_APPLICATION_ERROR過程,能夠從新定義異常錯誤消息,它爲應用程序提供了一種與ORACLE交互的方法。
RAISE_APPLICATION_ERROR 的語法以下:
RAISE_APPLICATION_ERROR(error_number,error_message,[keep_errors] ); 這裏的error_number 是從 –20,000 到 –20,999 之間的參數, error_message 是相應的提示信息(< 2048 字節),
keep_errors 爲可選,若是keep_errors =TRUE ,則新錯誤將被添加到已經引起的錯誤列表中。若是keep_errors=FALSE(缺省),則新錯誤將替換當前的錯誤列表。
例1:定義觸發器,使用RAISE_APPLICATION_ERROR阻止沒有員工姓名的新員式記錄插入:
CREATE OR REPLACE TRIGGER tr_insert_emp
BEFORE INSERT ON employees
FOR EACH ROW
BEGIN
IF :new.first_name IS NULL OR :new.last_name is null THEN
RAISE_APPLICATION_ERROR(-20000,'Employee must have a name.');
END IF;
END;
5.異常錯誤傳播
因爲異常錯誤能夠在聲明部分和執行部分以及異常錯誤部分出現,於是在不一樣部分引起的異常錯誤也不同。
在執行部分引起異常錯誤
當一個異常錯誤在執行部分引起時,有下列狀況:
l 若是當前塊對該異常錯誤設置了處理,則執行它併成功完成該塊的執行,而後控制轉給包含塊。
l 若是沒有對當前塊異常錯誤設置定義處理器,則經過在包含塊中引起它來傳播異常錯誤。而後對該包含塊執行步驟1)。
在聲明部分引起異常錯誤
若是在聲明部分引發異常狀況,即在聲明部分出現錯誤,那麼該錯誤就能影響到其它的塊。好比在有以下的PL/SQL程序:
DECLARE
name varchar2(12):='EricHu';
其它語句
BEGIN
其它語句
EXCEPTION
WHEN OTHERS THEN
其它語句
END;
異常錯誤處理編程
在通常的應用處理中,建議程序人員要用異常處理,由於若是程序中不聲明任何異常處理,則在程序運行出錯時,程序就被終止,而且也不提示任何信息。下面是使用系統提供的異常來編程的例子。
在 PL/SQL 中使用 SQLCODE, SQLERRM異常處理函數
因爲ORACLE 的錯信息最大長度是512字節,爲了獲得完整的錯誤提示信息,咱們可用 SQLERRM和 SUBSTR 函數一塊兒獲得錯誤提示信息,方便進行錯誤,特別是若是WHEN OTHERS異常處理器時更爲方便。
SQLCODE 返回遇到的Oracle錯誤號,
SQLERRM 返回遇到的Oracle錯誤信息.
如: SQLCODE=-100 è SQLERRM=’no_data_found ‘
SQLCODE=0 è SQLERRM=’normal, successfual completion’
例1. 將ORACLE錯誤代碼及其信息存入錯誤代碼表
CREATE TABLE errors (errnum NUMBER(4), errmsg VARCHAR2(100));
DECLARE
err_msg VARCHAR2(100);
BEGIN
/ 獲得全部 ORACLE 錯誤信息 /FOR err_num IN -100 .. 0 LOOPerr_msg := SQLERRM(err_num);INSERT INTO errors VALUES(err_num, err_msg);END LOOP;END;DROP TABLE errors;