Oracle PL/SQL語言初級教程之異常處理
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異常。
這裡replenish_inventory是一個觸發器。
處理異常
PL/SQL程式塊的異常部分包含了程式處理錯誤的程式碼,當異常被丟擲時,一個異常陷阱就自動發生,程式控制離開執行部分轉入異常部分,一旦程式進入異常部分就不能再回到同一塊的執行部分。下面是異常部分的一般語法:
使用者必須在獨立的WHEN子串中為每個異常設計異常處理程式碼,WHEN OTHERS子串必須放置在最後面作為預設處理器處理沒有顯式處理的異常。當異常發生時,控制轉到異常部分,ORACLE查詢當前異常相應的WHEN..THEN語句,捕捉異常,THEN之後的程式碼被執行,如果錯誤陷阱程式碼只是退出相應的巢狀塊,那麼程式將繼續執行內部塊END後面的語句。如果沒有找到相應的異常陷阱,那麼將執行WHEN OTHERS。在異常部分WHEN 子串沒有數量限制。
當異常丟擲後,控制無條件轉到異常部分,這就意味著控制不能回到異常發生的位置,當異常被處理和解決後,控制返回到上一層執行部分的下一條語句。
當異常發生時,在塊的內部沒有該異常處理器時,控制將轉到或傳播到上一層塊的異常處理部分。
異常傳播
沒有處理的異常將沿檢測異常呼叫程式傳播到外面,當異常被處理並解決或到達程式最外層傳播停止。
在宣告部分丟擲的異常將控制轉到上一層的異常部分。
執行部分丟擲的異常將首先傳遞到同一塊的異常部分,如果在同一塊的異常部分沒有處理這個異常的處理器,那麼異常將會傳播到上一層的異常部分中,一直到最外層。
在異常部分丟擲的異常將控制轉到上一層的異常部分。
處理異常將停止異常的傳播和解決。有時使用者希望在錯誤發生時,程式仍然能執行一些動作,要達到這個目的,可以把希望執行的動作放在異常處理器中,然後執行不帶引數的RAISE語句,RAISE語句將重新丟擲出現的異常,允許他繼續傳播。
如果從FORMAT_XXX_STACK輸出一個很大的值,那麼使用DBMS_OUTPUT或UTL_FILE顯示錯誤或呼叫堆的異常部分自身也會丟擲異常,這兩個堆常規下最多能返回2000位元組,但utl_file.put_line被限制在1000位元組以內,而dbms_output.put_line限制在512位元組內。如果使用前面的程式碼並且不允許這種可能性,那麼在異常處理器中將丟擲一個未處理的異常。
GOTO語句不能用於將控制從執行部分傳遞到異常部分或反之。[@more@]
當一個執行時錯誤發生時,稱為一個異常被丟擲。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/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- Go 語言異常處理Go
- Oracle 的PL/SQL語言使用OracleSQL
- oracle異常處理Oracle
- 【C++】 63_C語言異常處理C++C語言
- Oracle異常錯誤處理Oracle
- ORACLE 異常錯誤處理Oracle
- Oracle 監聽異常處理Oracle
- SQL Server 異常程式碼處理SQLServer
- oracle常見異常等待——latch處理思路Oracle
- 初學Solidity(六):Solidity異常處理Solid
- Oracle PL/SQLOracleSQL
- Oracle開發基礎-異常處理Oracle
- 異常篇——異常處理
- Oracle Gateway for SQL Server時2PC分散式事務異常處理OracleGatewaySQLServer分散式
- oracle分散式事務異常處理方法Oracle分散式
- 異常處理
- Oracle SQL處理OracleSQL
- 在 SQL Server 中使用 Try Catch 處理異常SQLServer
- Flutter 異常處理方案——灰度與降級Flutter
- C++語言程式設計筆記 - 第12章 - 異常處理C++程式設計筆記
- Go語言之Goroutine與通道、異常處理Go
- JSP 異常處理如何處理?JS
- 異常-throws的方式處理異常
- React 異常處理React
- JS異常處理JS
- Python——異常處理Python
- Python異常處理Python
- ThinkPHP 異常處理PHP
- JavaScript 異常處理JavaScript
- JAVA 異常處理Java
- 異常的處理
- golang - 異常處理Golang
- 異常處理2
- 異常處理1
- Java 異常處理Java
- Abp 異常處理
- JAVA異常處理Java
- 08、異常處理
- SpringMVC異常處理SpringMVC