Oracle PL/SQL語言初級教程之異常處理

hljhrbsjf發表於2006-06-26
 PL/SQL處理異常不同於其他程式語言的錯誤管理方法,PL/SQL的異常處理機制與ADA很相似,有一個處理錯誤的全包含方法。當發生錯誤時,程式無條件轉到異常處理部分,這就要求程式碼要非常乾淨並把錯誤處理部分和程式的其它部分分開。oracle允許宣告其他異常條件型別以擴充套件錯誤/異常處理。這種擴充套件使PL/SQL的異常處理非常靈活。

  當一個執行時錯誤發生時,稱為一個異常被丟擲。PL/SQL程式編譯時的錯誤不是能被處理得異常,只有在執行時的異常能被處理。在PL/SQL程式設計中異常的丟擲和處理是非常重要的內容。

  丟擲異常

  由三種方式丟擲異常

   . 透過PL/SQL執行時引擎

   . 使用RAISE語句

   . 呼叫RAISE_APPLICATION_ERROR儲存過程

  當資料庫或PL/SQL在執行時發生錯誤時,一個異常被PL/SQL執行時引擎自動丟擲。異常也可以透過RAISE語句丟擲

  RAISE exception_name;

  顯式丟擲異常是程式設計師處理宣告的異常的習慣用法,但RAISE不限於宣告瞭的異常,它可以丟擲任何任何異常。例如,你希望用TIMEOUT_ON_RESOURCE錯誤檢測新的執行時異常處理器,你只需簡單的在程式中使用下面的語句:

  RAISE TIMEOUT_ON_RESOUCE;

  下面看一個訂單輸入系統,當庫存小於訂單時丟擲一個inventory_too_low異常。

DECLARE
inventory_too_low EXCEPTION;
---其他宣告語句
BEGIN
.
.
IF order_rec.qty>inventory_rec.qty THEN
RAISE inventory_too_low;
END IF
.
.
EXCEPTION
WHEN inventory_too_low THEN
order_rec.staus:='backordered';
replenish_inventory(inventory_nbr=>
inventory_rec.sku,min_amount=>order_rec.qty-inventory_rec.qty);
END;

  這裡replenish_inventory是一個觸發器。
 
  處理異常

  PL/SQL程式塊的異常部分包含了程式處理錯誤的程式碼,當異常被丟擲時,一個異常陷阱就自動發生,程式控制離開執行部分轉入異常部分,一旦程式進入異常部分就不能再回到同一塊的執行部分。下面是異常部分的一般語法:

EXCEPTION
 WHEN exception_name THEN
  Code for handing exception_name
 [WHEN another_exception THEN
  Code for handing another_exception]
 [WHEN others THEN
  code for handing any other exception.]

  使用者必須在獨立的WHEN子串中為每個異常設計異常處理程式碼,WHEN OTHERS子串必須放置在最後面作為預設處理器處理沒有顯式處理的異常。當異常發生時,控制轉到異常部分,ORACLE查詢當前異常相應的WHEN..THEN語句,捕捉異常,THEN之後的程式碼被執行,如果錯誤陷阱程式碼只是退出相應的巢狀塊,那麼程式將繼續執行內部塊END後面的語句。如果沒有找到相應的異常陷阱,那麼將執行WHEN OTHERS。在異常部分WHEN 子串沒有數量限制。

EXCEPTION

 WHEN inventory_too_low THEN
  order_rec.staus:='backordered';
  replenish_inventory(inventory_nbr=>
  inventory_rec.sku,min_amount=>order_rec.qty-inventory_rec.qty);
 WHEN discontinued_item THEN
  --code for discontinued_item processing
 WHEN zero_divide THEN
  --code for zero_divide
 WHEN OTHERS THEN
  --code for any other exception
END;

  當異常丟擲後,控制無條件轉到異常部分,這就意味著控制不能回到異常發生的位置,當異常被處理和解決後,控制返回到上一層執行部分的下一條語句。

BEGIN
 DECLARE
  bad_credit;
 BEGIN
  RAISE bad_credit;
   --發生異常,控制轉向;
 EXCEPTION
  WHEN bad_credit THEN
   dbms_output.put_line('bad_credit');
  END;

  --bad_credit異常處理後,控制轉到這裡
 EXCEPTION
  WHEN OTHERS THEN
   --控制不會從bad_credit異常轉到這裡
   --因為bad_credit已被處理
END;


  當異常發生時,在塊的內部沒有該異常處理器時,控制將轉到或傳播到上一層塊的異常處理部分。

BEGIN
 DECLARE ---內部塊開始
  bad_credit;
 BEGIN
  RAISE bad_credit;
   --發生異常,控制轉向;
  EXCEPTION
  WHEN ZERO_DIVIDE THEN --不能處理bad_credite異常
   dbms_output.put_line('divide by zero error');
  END --結束內部塊

   --控制不能到達這裡,因為異常沒有解決;
   --異常部分

  EXCEPTION
  WHEN OTHERS THEN
   --由於bad_credit沒有解決,控制將轉到這裡
END;

  異常傳播

  沒有處理的異常將沿檢測異常呼叫程式傳播到外面,當異常被處理並解決或到達程式最外層傳播停止。
在宣告部分丟擲的異常將控制轉到上一層的異常部分。

BEGIN
executable statements
BEGIN
today DATE:='SYADATE'; --ERRROR
BEGIN --內部塊開始
dbms_output.put_line('this line will not execute');
EXCEPTION
WHEN OTHERS THEN
--異常不會在這裡處理
END;--內部塊結束

EXCEPTION
WHEN OTHERS THEN
處理異常
END

  執行部分丟擲的異常將首先傳遞到同一塊的異常部分,如果在同一塊的異常部分沒有處理這個異常的處理器,那麼異常將會傳播到上一層的異常部分中,一直到最外層。

  在異常部分丟擲的異常將控制轉到上一層的異常部分。

  處理異常將停止異常的傳播和解決。有時使用者希望在錯誤發生時,程式仍然能執行一些動作,要達到這個目的,可以把希望執行的動作放在異常處理器中,然後執行不帶引數的RAISE語句,RAISE語句將重新丟擲出現的異常,允許他繼續傳播。

DECLARE
order_too_old EXCEPTION;
BEGIN
RAISE order_too_old;
EXCEPTION
WHEN order_too_old THEN
DECLARE
file_handle UTL_FILE.FILE_TYPE;
BEGIN
--open file
file_handle:=UTL_FILE.FOPEN
(location=>'/ora01/app/oracle/admin/test/utlsir'
,filename=>'error.log'
.open_mode=>'W');
--write error stack
UTL_FILE.PUT_LINE(filehandle,
DBMS_UTILITY.FORMAT_ERROR_STACK);
--write the call stack
UTL_FILE.PUT_LINE(filehandle,
DBMS_UTILITY.FORMAT_CALL_STACK);
--close error log
UTL_FILE.FCLOSE(file_handle);
RAISE; --re-raise the exception
END
END

  如果從FORMAT_XXX_STACK輸出一個很大的值,那麼使用DBMS_OUTPUT或UTL_FILE顯示錯誤或呼叫堆的異常部分自身也會丟擲異常,這兩個堆常規下最多能返回2000位元組,但utl_file.put_line被限制在1000位元組以內,而dbms_output.put_line限制在512位元組內。如果使用前面的程式碼並且不允許這種可能性,那麼在異常處理器中將丟擲一個未處理的異常。

  GOTO語句不能用於將控制從執行部分傳遞到異常部分或反之。[@more@]

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/631872/viewspace-845709/,如需轉載,請註明出處,否則將追究法律責任。

相關文章