總結:整理 oracle異常錯誤處理 .
5.1 異常處理概念
5.1.1 預定義的異常處理
5.1.2 非預定義的異常處理
5.1.3 使用者自定義的異常處理
5.1.4
5.2 異常錯誤傳播
5.2.1 在執行部分引發異常錯誤
5.2.2 在宣告部分引發異常錯誤
5.3 異常錯誤處理程式設計
5.4
即使是寫得最好的PL/SQL程式也會遇到錯誤或未預料到的事件。一個優秀的程式都應該能夠正確處理各種出錯情況,並儘可能從錯誤中恢復。任何ORACLE錯誤(報告為ORA-xxxxx
ORACLE 提供異常情況(EXCEPTION)和異常處理(EXCEPTION HANDLER)來實現錯誤處理。
異常情況處理(EXCEPTION)是用來處理正常執行過程中未預料的事件,程式塊的異常處理預定義的錯誤和自定義錯誤,由於PL/SQL程式塊一旦產生異常而沒有指出如何處理時,程式就會自動終止整個程式執行.
有三種類型的異常錯誤:
程式執行過程中,出現程式設計人員認為的非正常情況。對這種異常情況的處理,需要使用者在程式中定義,然後顯式地在程式中將其引發。
異常處理部分一般放在 PL/SQL 程式體的後半部,結構為:
EXCEPTION
WHEN second_exception THEN
WHEN OTHERS THEN
END;
異常處理可以按任意次序排列,但 OTHERS 必須放在最後.
5.1.1 預定義的異常處理
錯誤號 |
異常錯誤資訊名稱 |
說明 |
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 索引得引用大於集合中元素的個數. |
對這種異常情況的處理,只需在PL/SQL塊的異常處理部分,直接引用相應的異常情況名,並對其完成相應的異常錯誤處理即可。
例1:更新指定員工工資,如工資小於1500,則加100;
DECLARE
v_empno employees.employee_id%TYPE := &empno;
v_sal employees.salary%TYPE;
BEGIN
SELECT salary INTO v_sal FROM employees WHERE employee_id = v_empno;
IF v_sal<=1500 THEN
UPDATE employees
SET salary = salary + 100 WHERE employee_id=v_empno;
DBMS_OUTPUT.PUT_LINE('編碼為'||v_empno||'員工工資已更新!');
ELSE
DBMS_OUTPUT.PUT_LINE('編碼為'||v_empno||'員工工資已經超過規定值!');
END IF;
EXCEPTION
WHEN NO_DATA_FOUND THEN
DBMS_OUTPUT.PUT_LINE('資料庫中沒有編碼為'||v_empno||'的員工');
WHEN TOO_MANY_ROWS THEN
DBMS_OUTPUT.PUT_LINE('程式執行錯誤!請使用遊標');
WHEN OTHERS THEN
DBMS_OUTPUT.PUT_LINE(SQLCODE||'---'||SQLERRM);
END;
BEGIN
EXCEPTION
END;
對於這類異常情況的處理,首先必須對非定義的ORACLE錯誤進行定義。步驟如下:
1. 在PL/SQL 塊的定義部分定義異常情況:
<異常情況>2. 將其定義好的異常情況,與標準的ORACLE錯誤聯絡起來,使用EXCEPTION_INIT語句:
PRAGMA EXCEPTION_INIT(<異常情況>, <錯誤程式碼>);3. 在PL/SQL 塊的異常情況處理部分對異常情況做出相應的處理。
例2:刪除指定部門的記錄資訊,以確保該部門沒有員工。
INSERT INTO departments VALUES(50, 'FINANCE', 'CHICAGO');DECLARE
BEGIN
EXCEPTION
END;
當與一個異常錯誤相關的錯誤出現時,就會隱含觸發該異常錯誤。使用者定義的異常錯誤是通過顯式使用 RAISE 語句來觸發。當引發一個異常錯誤時,控制就轉向到 EXCEPTION塊異常錯誤部分,執行錯誤處理程式碼。
對於這類異常情況的處理,步驟如下:
1. 在PL/SQL 塊的定義部分定義異常情況:
<異常情況>2. RAISE <異常情況>;
3. 在PL/SQL 塊的異常情況處理部分對異常情況做出相應的處理。
例3:更新指定員工工資,增加100;
DECLAREBEGIN
EXCEPTION
END;
呼叫DBMS_STANDARD(ORACLE提供的包)包所定義的RAISE_APPLICATION_ERROR過程,可以重新定義異常錯誤訊息,它為應用程式提供了一種與ORACLE互動的方法。
RAISE_APPLICATION_ERROR 的語法如下:
keep_errors 為可選,如果keep_errors =TRUE ,則新錯誤將被新增到已經引發的錯誤列表中。如果keep_errors=FALSE(預設),則新錯誤將替換當前的錯誤列表。
例4:建立一個函式get_salary, 該函式檢索指定部門的工資總和,其中定義了-20991和-20992號錯誤,分別處理引數為空和非法部門程式碼兩種錯誤:
CREATE TABLE errlog(CREATE OR REPLACE FUNCTION get_salary(p_deptno NUMBER)
RETURN NUMBER
AS
BEGIN
END;
DECLARE
BEGIN
END outer;
例5:定義觸發器,使用RAISE_APPLICATION_ERROR阻止沒有員工姓名的新員式記錄插入:
CREATE OR REPLACE TRIGGER tr_insert_empBEFORE INSERT ON employees
FOR EACH ROW
BEGIN
END;
l 如果當前塊對該異常錯誤設定了處理,則執行它併成功完成該塊的執行,然後控制轉給包含塊。
l 如果沒有對當前塊異常錯誤設定定義處理器,則通過在包含塊中引發它來傳播異常錯誤。然後對該包含塊執行步驟1)。
BEGIN
EXCEPTION
END;
EXCEPTION
WHEN OTHERS THEN
END;
SQLCODE
SQLERRM
如:
SQLCODE=0
例6. 將ORACLE錯誤程式碼及其資訊存入錯誤程式碼表
CREATE TABLE errors (errnum NUMBER(4), errmsg VARCHAR2(100));DECLARE
BEGIN
END;
DROP TABLE errors;
例7. 查詢ORACLE錯誤程式碼;
BEGINEXCEPTION
END;
例8. 利用ORACLE錯誤程式碼,編寫異常錯誤處理程式碼;
DECLARE