<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    kxbin
    成功留給有準備的人
    posts - 10,  comments - 35,  trackbacks - 0

    本篇主要內容如下:

    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 中使用 SQLCODE, SQLERRM異常處理函數


      即使是寫得最好的PL/SQL程序也會遇到錯誤或未預料到的事件一個優秀的程序都應該能夠正確處理各種出錯情況,并盡可能從錯誤中恢復。任何ORACLE錯誤(報告為ORA-xxxxx形式的Oracle錯誤號)、PL/SQL運行錯誤或用戶定義條件(不一寫是錯誤),都可以。當然了,PL/SQL編譯錯誤不能通過PL/SQL異常處理來處理,因為這些錯誤發生在PL/SQL程序執行之前。

    ORACLE 提供異常情況(EXCEPTION)和異常處理(EXCEPTION HANDLER)來實現錯誤處理。

     

    5.1 異常處理概念

    異常情況處理(EXCEPTION)是用來處理正常執行過程中未預料的事件,程序塊的異常處理預定義的錯誤和自定義錯誤,由于PL/SQL程序塊一旦產生異常而沒有指出如何處理時,程序就會自動終止整個程序運行.

    有三種類型的異常錯誤:

        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 必須放在最后.

    5.1.1 預定義的異常處理

       預定義說明的部分 ORACLE 異常錯誤

    錯誤號

    異常錯誤信息名稱

    說明

    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
    復制代碼

     5.1.2 非預定義的異常處理

    對于這類異常情況的處理,首先必須對非定義的ORACLE錯誤進行定義。步驟如下:

    1. 在PL/SQL 塊的定義部分定義異常情況:

     

    <異常情況>  EXCEPTION;

     

    2. 將其定義好的異常情況,與標準的ORACLE錯誤聯系起來,使用EXCEPTION_INIT語句:

    PRAGMA EXCEPTION_INIT(<異常情況><錯誤代碼>);

     

    3. 在PL/SQL 塊的異常情況處理部分對異常情況做出相應的處理。

     

    例2:刪除指定部門的記錄信息,以確保該部門沒有員工。

     

    復制代碼
    INSERT INTO departments VALUES(50'FINANCE''CHICAGO');

    DECLARE
       v_deptno departments.department_id
    %TYPE := &deptno;
       deptno_remaining EXCEPTION;
       PRAGMA EXCEPTION_INIT(deptno_remaining, 
    -2292);
       
    /* -2292 是違反一致性約束的錯誤代碼 */
    BEGIN
       
    DELETE FROM departments WHERE department_id = v_deptno;
    EXCEPTION
       
    WHEN deptno_remaining THEN 
          DBMS_OUTPUT.PUT_LINE(
    '違反數據完整性約束!');
       
    WHEN OTHERS THEN
          DBMS_OUTPUT.PUT_LINE(SQLCODE
    ||'---'||SQLERRM);
    END;
    復制代碼

     

     

    5.1.3 用戶自定義的異常處理

    當與一個異常錯誤相關的錯誤出現時,就會隱含觸發該異常錯誤。用戶定義的異常錯誤是通過顯式使用 RAISE 語句來觸發。當引發一個異常錯誤時,控制就轉向到 EXCEPTION塊異常錯誤部分,執行錯誤處理代碼。

    對于這類異常情況的處理,步驟如下:

    1. 在PL/SQL 塊的定義部分定義異常情況:

     

    <異常情況>  EXCEPTION;

     

    2. RAISE <異常情況>;

     

    3. 在PL/SQL 塊的異常情況處理部分對異常情況做出相應的處理。

     

    例3:更新指定員工工資,增加100;

     

    復制代碼
    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
          RAISE no_result;
       
    END IF;
    EXCEPTION
       
    WHEN no_result THEN 
          DBMS_OUTPUT.PUT_LINE(
    '你的數據更新語句失敗了!');
       
    WHEN OTHERS THEN
          DBMS_OUTPUT.PUT_LINE(SQLCODE
    ||'---'||SQLERRM);
    END;
    復制代碼

     

     

    5.1.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(缺省),則新錯誤將替換當前的錯誤列表。

    例4:創建一個函數get_salary, 該函數檢索指定部門的工資總和,其中定義了-20991和-20992號錯誤,分別處理參數為空和非法部門代碼兩種錯誤:

     

    復制代碼
    CREATE TABLE errlog(
      Errcode 
    NUMBER,
      Errtext 
    CHAR(40));

    CREATE OR REPLACE FUNCTION get_salary(p_deptno NUMBER)
    RETURN NUMBER 
    AS
      v_sal 
    NUMBER;
    BEGIN
      
    IF p_deptno IS NULL THEN
        RAISE_APPLICATION_ERROR(
    -20991, ’部門代碼為空’);
      ELSIF p_deptno
    <0 THEN
        RAISE_APPLICATION_ERROR(
    -20992, ’無效的部門代碼’);
      
    ELSE
        
    SELECT SUM(employees.salary) INTO v_sal FROM employees 
        
    WHERE employees.department_id=p_deptno;
        
    RETURN v_sal;
      
    END IF;
    END;

    DECLARE 
      V_salary 
    NUMBER(7,2);
      V_sqlcode 
    NUMBER;
      V_sqlerr 
    VARCHAR2(512);
      Null_deptno EXCEPTION;
      Invalid_deptno EXCEPTION;
      PRAGMA EXCEPTION_INIT(null_deptno,
    -20991);
      PRAGMA EXCEPTION_INIT(invalid_deptno, 
    -20992);
    BEGIN
      V_salary :
    =get_salary(10);
      DBMS_OUTPUT.PUT_LINE(
    '10號部門工資:' || TO_CHAR(V_salary));

      
    BEGIN
        V_salary :
    =get_salary(-10);
      EXCEPTION
        
    WHEN invalid_deptno THEN
          V_sqlcode :
    =SQLCODE;
          V_sqlerr  :
    =SQLERRM;
          
    INSERT INTO errlog(errcode, errtext) 
          
    VALUES(v_sqlcode, v_sqlerr);
          
    COMMIT;
      
    END inner1;

      V_salary :
    =get_salary(20);
      DBMS_OUTPUT.PUT_LINE(
    '部門號為20的工資為:'||TO_CHAR(V_salary));

      
    BEGIN
        V_salary :
    =get_salary(NULL);
      
    END inner2;

      V_salary :
    = get_salary(30);
      DBMS_OUTPUT.PUT_LINE(
    '部門號為30的工資為:'||TO_CHAR(V_salary));

      EXCEPTION
        
    WHEN null_deptno THEN
          V_sqlcode :
    =SQLCODE;
          V_sqlerr  :
    =SQLERRM;
          
    INSERT INTO errlog(errcode, errtext) VALUES(v_sqlcode, v_sqlerr);
          
    COMMIT;
        
    WHEN OTHERS THEN
             DBMS_OUTPUT.PUT_LINE(SQLCODE
    ||'---'||SQLERRM);
    END outer;
    復制代碼

     

     

    5定義觸發器,使用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.2 異常錯誤傳播

        由于異常錯誤可以在聲明部分和執行部分以及異常錯誤部分出現,因而在不同部分引發的異常錯誤也不一樣。

     

    5.2.1 在執行部分引發異常錯誤

        當一個異常錯誤在執行部分引發時,有下列情況:

    如果當前塊對該異常錯誤設置了處理,則執行它并成功完成該塊的執行,然后控制轉給包含塊。

    如果沒有對當前塊異常錯誤設置定義處理器,則通過在包含塊中引發它來傳播異常錯誤。然后對該包含塊執行步驟1)。

     

    5.2.2 在聲明部分引發異常錯誤

        如果在聲明部分引起異常情況,即在聲明部分出現錯誤,那么該錯誤就能影響到其它的塊。比如在有如下的PL/SQL程序:

    復制代碼
    DECLARE
        name 
    varchar2(12):='EricHu';
        其它語句
    BEGIN
        其它語句
    EXCEPTION
        
    WHEN OTHERS THEN 
        其它語句
    END;
    復制代碼

     

         例子中,由于Abc number(3)=’abc’; 出錯,盡管在EXCEPTION中說明了WHEN OTHERS THEN語句,但WHEN OTHERS THEN也不會被執行。 但是如果在該錯誤語句塊的外部有一個異常錯誤,則該錯誤能被抓住,如:

     

    復制代碼
    BEGIN
        
    DECLARE
        name 
    varchar2(12):='EricHu';
        其它語句
       
    BEGIN
        其它語句
       EXCEPTION
        
    WHEN OTHERS THEN 
        其它語句
        
    END;
    EXCEPTION
    WHEN OTHERS THEN 
        其它語句
    END;
    復制代碼

     

     

    5.3 異常錯誤處理編程

        在一般的應用處理中,建議程序人員要用異常處理,因為如果程序中不聲明任何異常處理,則在程序運行出錯時,程序就被終止,并且也不提示任何信息。下面是使用系統提供的異常來編程的例子。

     

    5.4  在 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’

    6將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查詢ORACLE錯誤代碼;

     

    復制代碼
    BEGIN
       
    INSERT INTO employees(employee_id, first_name,last_name,hire_date,department_id)
       
    VALUES(2222'Eric','Hu', SYSDATE, 20);
       DBMS_OUTPUT.PUT_LINE(
    '插入數據記錄成功!');
       
       
    INSERT INTO employees(employee_id, first_name,last_name,hire_date,department_id)
       
    VALUES(2222'','', SYSDATE, 20);
       DBMS_OUTPUT.PUT_LINE(
    '插入數據記錄成功!');
    EXCEPTION
       
    WHEN OTHERS THEN
          DBMS_OUTPUT.PUT_LINE(SQLCODE
    ||'---'||SQLERRM);
    END;
    復制代碼

     

     

    8利用ORACLE錯誤代碼,編寫異常錯誤處理代碼;

     

    復制代碼
    DECLARE
       empno_remaining EXCEPTION;
       PRAGMA EXCEPTION_INIT(empno_remaining, 
    -1);
       
    /* -1 是違反唯一約束條件的錯誤代碼 */
    BEGIN
       
    INSERT INTO employees(employee_id, first_name,last_name,hire_date,department_id)
       
    VALUES(3333'Eric','Hu', SYSDATE, 20);
       DBMS_OUTPUT.PUT_LINE(
    '插入數據記錄成功!');
       
       
    INSERT INTO employees(employee_id, first_name,last_name,hire_date,department_id)
       
    VALUES(3333'','',SYSDATE, 20);
       DBMS_OUTPUT.PUT_LINE(
    '插入數據記錄成功!');
    EXCEPTION
       
    WHEN empno_remaining THEN 
          DBMS_OUTPUT.PUT_LINE(
    '違反數據完整性約束!');
       
    WHEN OTHERS THEN
          DBMS_OUTPUT.PUT_LINE(SQLCODE
    ||'---'||SQLERRM);
    END;
    posted on 2012-09-12 17:32 kxbin 閱讀(584) 評論(0)  編輯  收藏 所屬分類: ORACLE轉發
    你恨一個人是因為你愛他;你喜歡一個人,是因為他身上有你沒有的;你討厭一個人是因為他身上有你有的東西;你經常在別人面前批評某人,其實潛意識中是想接近他。

    <2025年5月>
    27282930123
    45678910
    11121314151617
    18192021222324
    25262728293031
    1234567

    常用鏈接

    留言簿(5)

    隨筆檔案

    文章分類

    文章檔案

    相冊

    收藏夾

    J2EE

    java技術網站

    Linux

    平時常去的網站

    數據庫

    電影網站

    網站設計

    搜索

    •  

    最新評論

    閱讀排行榜

    評論排行榜

    主站蜘蛛池模板: 亚洲一区二区三区免费视频| 女性自慰aⅴ片高清免费| 亚洲欧洲国产精品久久| 成人免费看片又大又黄| EEUSS影院WWW在线观看免费| 久久亚洲精品无码AV红樱桃| 日韩精品免费一区二区三区| 中文字幕在线免费观看视频| 99999久久久久久亚洲| 国产性爱在线观看亚洲黄色一级片| 18级成人毛片免费观看| 黄色a级片免费看| 亚洲日韩国产精品无码av| 国产精品极品美女免费观看| 免费在线观看一级片| 九九精品国产亚洲AV日韩| 久久青青草原亚洲AV无码麻豆| 免费无码又爽又高潮视频| 无码一区二区三区免费| 在线91精品亚洲网站精品成人| 中文字幕亚洲综合精品一区| 亚洲精品成人片在线观看| 青青青国产在线观看免费网站| 国产在线观看免费av站| 亚洲AV无码国产精品永久一区| 91亚洲一区二区在线观看不卡| 亚洲午夜精品第一区二区8050| 久久WWW免费人成人片| 国产高清不卡免费视频| 一区二区三区AV高清免费波多| 四虎必出精品亚洲高清| 亚洲AV日韩精品久久久久久久| 亚洲国产成人久久综合碰| 免费a级毛片无码a∨蜜芽试看| 免费在线看黄网站| 人成电影网在线观看免费| 亚洲AV日韩AV一区二区三曲| 亚洲无成人网77777| 亚洲va国产va天堂va久久| 亚洲伊人成无码综合网| 国产成人精品123区免费视频|