即使是寫(xiě)得最好的PL/SQL程序也會(huì)遇到錯(cuò)誤或未預(yù)料到的事件。一個(gè)優(yōu)秀的程序都應(yīng)該能夠正確處理各種出錯(cuò)情況,并盡可能從錯(cuò)誤中恢復(fù)。任何ORACLE錯(cuò)誤(報(bào)告為ORA-xxxxx形式的Oracle錯(cuò)誤號(hào))、PL/SQL運(yùn)行錯(cuò)誤或用戶定義條件(不一寫(xiě)是錯(cuò)誤),都可以。當(dāng)然了,PL/SQL編譯錯(cuò)誤不能通過(guò)PL/SQL異常處理來(lái)處理,因?yàn)檫@些錯(cuò)誤發(fā)生在PL/SQL程序執(zhí)行之前。
ORACLE 提供異常情況(EXCEPTION)和異常處理(EXCEPTION HANDLER)來(lái)實(shí)現(xiàn)錯(cuò)誤處理。
成都創(chuàng)新互聯(lián)主要從事網(wǎng)站制作、做網(wǎng)站、網(wǎng)頁(yè)設(shè)計(jì)、企業(yè)做網(wǎng)站、公司建網(wǎng)站等業(yè)務(wù)。立足成都服務(wù)蘭陵,十載網(wǎng)站建設(shè)經(jīng)驗(yàn),價(jià)格優(yōu)惠、服務(wù)專業(yè),歡迎來(lái)電咨詢建站服務(wù):18980820575
1 異常處理概念
異常情況處理(EXCEPTION)是用來(lái)處理正常執(zhí)行過(guò)程中未預(yù)料的事件,程序塊的異常處理預(yù)定義的錯(cuò)誤和自定義錯(cuò)誤,由于PL/SQL程序塊一旦產(chǎn)生異常而沒(méi)有指出如何處理時(shí),程序就會(huì)自動(dòng)終止整個(gè)程序運(yùn)行.
有三種類型的異常錯(cuò)誤:
1. 預(yù)定義 ( Predefined )錯(cuò)誤
ORACLE預(yù)定義的異常情況大約有24個(gè)。對(duì)這種異常情況的處理,無(wú)需在程序中定義,由ORACLE自動(dòng)將其引發(fā)。
2. 非預(yù)定義 ( Predefined )錯(cuò)誤
即其他標(biāo)準(zhǔn)的ORACLE錯(cuò)誤。對(duì)這種異常情況的處理,需要用戶在程序中定義,然后由ORACLE自動(dòng)將其引發(fā)。
3. 用戶定義(User_define) 錯(cuò)誤
程序執(zhí)行過(guò)程中,出現(xiàn)編程人員認(rèn)為的非正常情況。對(duì)這種異常情況的處理,需要用戶在程序中定義,然后顯式地在程序中將其引發(fā)。
異常處理部分一般放在 PL/SQL 程序體的后半部,結(jié)構(gòu)為:
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 必須放在最后.
1 預(yù)定義的異常處理
預(yù)定義說(shuō)明的部分 ORACLE 異常錯(cuò)誤
所有錯(cuò)誤代碼和描述,文件在$ORACLE_HOME/rdbms/mesg/oraus.msg
對(duì)這種異常情況的處理,只需在PL/SQL塊的異常處理部分,直接引用相應(yīng)的異常情況名,并對(duì)其完成相應(yīng)的異常錯(cuò)誤處理即可。
例1:更新指定員工工資,如工資小于1500,則加100;
D:\instantclient_12_2>sqlplus hr/hr@pdbtest
SQL*Plus: Release 12.2.0.1.0 Production on Thu Dec 21 09:55:15 2017
Copyright (c) 1982, 2016, Oracle. All rights reserved.
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||'員工工資已經(jīng)超過(guò)規(guī)定值!');
11 END IF;
12 EXCEPTION
13 WHEN NO_DATA_FOUND THEN
14 DBMS_OUTPUT.PUT_LINE('數(shù)據(jù)庫(kù)中沒(méi)有編碼為'||v_empno||'的員工');
15 WHEN TOO_MANY_ROWS THEN
16 DBMS_OUTPUT.PUT_LINE('程序運(yùn)行錯(cuò)誤!請(qǐng)使用游標(biāo)');
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;
數(shù)據(jù)庫(kù)中沒(méi)有編碼為7782的員工
PL/SQL procedure successfully completed.
2 非預(yù)定義的異常處理
對(duì)于這類異常情況的處理,首先必須對(duì)非定義的ORACLE錯(cuò)誤進(jìn)行定義。步驟如下:
<異常情況> EXCEPTION;
將其定義好的異常情況,與標(biāo)準(zhǔn)的ORACLE錯(cuò)誤聯(lián)系起來(lái),使用EXCEPTION_INIT語(yǔ)句:
PRAGMA EXCEPTION_INIT(<異常情況>, <錯(cuò)誤代碼>);
例1:刪除指定部門的記錄信息,以確保該部門沒(méi)有員工。
SQL> DECLARE
2 v_deptno departments.department_id%TYPE := &deptno;
3 deptno_remaining EXCEPTION;
4 PRAGMA EXCEPTION_INIT(deptno_remaining, -2292);
5 / -2292 是違反一致性約束的錯(cuò)誤代碼 /
6 BEGIN
7 DELETE FROM departments WHERE department_id = v_deptno;
8 EXCEPTION
9 WHEN deptno_remaining THEN
10 DBMS_OUTPUT.PUT_LINE('違反數(shù)據(jù)完整性約束!');
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;
違反數(shù)據(jù)完整性約束!
PL/SQL procedure successfully completed.
3 用戶自定義的異常處理
當(dāng)與一個(gè)異常錯(cuò)誤相關(guān)的錯(cuò)誤出現(xiàn)時(shí),就會(huì)隱含觸發(fā)該異常錯(cuò)誤。用戶定義的異常錯(cuò)誤是通過(guò)顯式使用 RAISE 語(yǔ)句來(lái)觸發(fā)。當(dāng)引發(fā)一個(gè)異常錯(cuò)誤時(shí),控制就轉(zhuǎn)向到 EXCEPTION塊異常錯(cuò)誤部分,執(zhí)行錯(cuò)誤處理代碼。
對(duì)于這類異常情況的處理,步驟如下:
1. 在PL/SQL 塊的定義部分定義異常情況:
<異常情況> EXCEPTION;
2. RAISE <異常情況>;
3. 在PL/SQL 塊的異常情況處理部分對(duì)異常情況做出相應(yīng)的處理。
例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('你的數(shù)據(jù)更新語(yǔ)句失敗了!');
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;
你的數(shù)據(jù)更新語(yǔ)句失敗了!
PL/SQL procedure successfully completed.
4 用戶定義的異常處理
調(diào)用DBMS_STANDARD(ORACLE提供的包)包所定義的RAISE_APPLICATION_ERROR過(guò)程,可以重新定義異常錯(cuò)誤消息,它為應(yīng)用程序提供了一種與ORACLE交互的方法。
RAISE_APPLICATION_ERROR 的語(yǔ)法如下:
RAISE_APPLICATION_ERROR(error_number,error_message,[keep_errors] );
這里的error_number 是從 –20,000 到 –20,999 之間的參數(shù),
error_message 是相應(yīng)的提示信息(< 2048 字節(jié)),
keep_errors 為可選,如果keep_errors =TRUE ,則新錯(cuò)誤將被添加到已經(jīng)引發(fā)的錯(cuò)誤列表中。如果keep_errors=FALSE(缺省),則新錯(cuò)誤將替換當(dāng)前的錯(cuò)誤列表。
例1:定義觸發(fā)器,使用RAISE_APPLICATION_ERROR阻止沒(méi)有員工姓名的新員式記錄插入:
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.異常錯(cuò)誤傳播
由于異常錯(cuò)誤可以在聲明部分和執(zhí)行部分以及異常錯(cuò)誤部分出現(xiàn),因而在不同部分引發(fā)的異常錯(cuò)誤也不一樣。
在執(zhí)行部分引發(fā)異常錯(cuò)誤
當(dāng)一個(gè)異常錯(cuò)誤在執(zhí)行部分引發(fā)時(shí),有下列情況:
l 如果當(dāng)前塊對(duì)該異常錯(cuò)誤設(shè)置了處理,則執(zhí)行它并成功完成該塊的執(zhí)行,然后控制轉(zhuǎn)給包含塊。
l 如果沒(méi)有對(duì)當(dāng)前塊異常錯(cuò)誤設(shè)置定義處理器,則通過(guò)在包含塊中引發(fā)它來(lái)傳播異常錯(cuò)誤。然后對(duì)該包含塊執(zhí)行步驟1)。
在聲明部分引發(fā)異常錯(cuò)誤
如果在聲明部分引起異常情況,即在聲明部分出現(xiàn)錯(cuò)誤,那么該錯(cuò)誤就能影響到其它的塊。比如在有如下的PL/SQL程序:
DECLARE
name varchar2(12):='EricHu';
其它語(yǔ)句
BEGIN
其它語(yǔ)句
EXCEPTION
WHEN OTHERS THEN
其它語(yǔ)句
END;
異常錯(cuò)誤處理編程
在一般的應(yīng)用處理中,建議程序人員要用異常處理,因?yàn)槿绻绦蛑胁宦暶魅魏萎惓L幚恚瑒t在程序運(yùn)行出錯(cuò)時(shí),程序就被終止,并且也不提示任何信息。下面是使用系統(tǒng)提供的異常來(lái)編程的例子。
在 PL/SQL 中使用 SQLCODE, SQLERRM異常處理函數(shù)
由于ORACLE 的錯(cuò)信息最大長(zhǎng)度是512字節(jié),為了得到完整的錯(cuò)誤提示信息,我們可用 SQLERRM和 SUBSTR 函數(shù)一起得到錯(cuò)誤提示信息,方便進(jìn)行錯(cuò)誤,特別是如果WHEN OTHERS異常處理器時(shí)更為方便。
SQLCODE 返回遇到的Oracle錯(cuò)誤號(hào),
SQLERRM 返回遇到的Oracle錯(cuò)誤信息.
如: SQLCODE=-100 è SQLERRM=’no_data_found ‘
SQLCODE=0 è SQLERRM=’normal, successfual completion’
例1. 將ORACLE錯(cuò)誤代碼及其信息存入錯(cuò)誤代碼表
CREATE TABLE errors (errnum NUMBER(4), errmsg VARCHAR2(100));
DECLARE
err_msg VARCHAR2(100);
BEGIN
/ 得到所有 ORACLE 錯(cuò)誤信息 /
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;
當(dāng)前標(biāo)題:ORACLE異常錯(cuò)誤處理
地址分享:http://www.rwnh.cn/article36/jcjjsg.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站制作、網(wǎng)站策劃、品牌網(wǎng)站設(shè)計(jì)、虛擬主機(jī)、云服務(wù)器、全網(wǎng)營(yíng)銷推廣
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)