1、異常概述
異常(Exception)是一種錯誤處理機制,用于在指定的錯誤發生時改變腳本的正常流程。
當異常被觸發時,當前代碼狀態被保存,代碼執行被切換到預定義的異常處理器函數(如果有)
根據情況,處理器也許會從保存的代碼狀態重新開始執行代碼,終止腳本執行,或從代碼中另外的位置繼續執行腳本
2、異常的基本使用
當異常被拋出時,其后的代碼不會繼續執行,PHP 會嘗試查找匹配的 "catch" 代碼塊。
如果異常沒有被捕獲,而且又沒用使用 set_exception_handler() 作相應的處理的話,那么將發生一個嚴重的錯誤(致命錯誤),并且輸出 "Uncaught Exception" (未捕獲異常)的錯誤消息。
嘗試拋出一個異常,同時不去捕獲它
1
2
3
4
5
6
7
8
|
//create function with an exception function checkNum( $number ){ if ( $number >1){ throw new Exception( "Value must be 1 or below" ); } } //trigger exception checkNum(2); |
上面的代碼會獲得類似這樣的一個錯誤:
Fatal error: Uncaught exception 'Exception' with message 'Value must be 1 or below' in C:\webfolder\test.php:6
Stack trace: #0 C:\webfolder\test.php(12):checkNum(28) #1 {main} thrown in C:\webfolder\test.php on line 6
注意:PHP默認是警告模式,如果需要對系統錯誤使用異常處理機制,則要在處理代碼之前設置錯誤處理模式
1
2
3
|
set_error_handler( function (){ throw new Exception( '錯誤!' ); }); |
3、Try, throw 和 catch
正確的處理程序應當包括:
Try - 使用異常的函數應該位于 "try" 代碼塊內。如果沒有觸發異常,則代碼將照常繼續執行。但是如果異常被觸發,會拋出一個異常。
Throw - 這里規定如何觸發異常。每一個 "throw" 必須對應至少一個 "catch"
Catch - "catch" 代碼塊會捕獲異常,并創建一個包含異常信息的對象
1
2
3
4
5
6
7
8
9
10
11
12
13
|
//創建可拋出一個異常的函數 function checkNum( $number ){ if ( $number >1){ throw new Exception( "Value must be 1 or below" ); } } //在 "try" 代碼塊中觸發異常 try { checkNum(2); } catch (Exception $e ) { //捕獲異常 echo 'File: ' . $e ->getFile(). ' line: ' . $e ->getLine(). '<br>' ; die ( 'Message: ' . $e ->getMessage()); } |
上面代碼將獲得類似這樣一個錯誤:
File: E:\webdev\www\pdo\3.php line: 7
Message: Value must be 1 or below
代碼解析:創建 checkNum() 函數。它檢測數字是否大于 1。如果是,則拋出一個異常。在 "try" 代碼塊中調用 checkNum(),函數checkNum() 函數中的異常被拋出,"catch" 代碼塊接收到該異常,并創建一個包含異常信息的對象 ($e)。通過從這個 exception 對象輸出來自該異常的錯誤消息
不過,為了遵循“每個 throw 必須對應一個 catch”的原則,可以設置一個頂層的異常處理器來處理漏掉的錯誤,見第7點
4、創建自定義的異常處理器
創建一個專門的類,當 PHP 中發生異常時,可調用其函數。該類必須是 exception 類的一個擴展。這個自定義的exception 類繼承了 PHP 的 exception 類的所有屬性,可向其添加自定義的函數。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
class customException extends Exception{ public function errorMessage(){ return 'Error on line ' . $this ->getLine(). ' in ' . $this ->getFile(). ': <b>' . $this ->getMessage(). '</b> is not a valid E-Mail address' ; } } try { //使用PHP過濾器驗證郵箱有效性 if (filter_var( $email , FILTER_VALIDATE_EMAIL) === FALSE){ throw new customException( $email ); } } catch (customException $e ){ echo $e ->errorMessage(); } |
這個新的類是舊的 exception 類的副本,外加 errorMessage() 函數。正因為它是舊類的副本,因此它從舊類繼承了屬性和方法,我們可以使用 exception 類的方法,比如 getLine() 、 getFile() 以及 getMessage()。
上面的代碼拋出了一個異常,并通過一個自定義的 exception 類來捕獲它:創建 errorMessage() 函數。如果 e-mail 地址不合法,則該函數返回一條錯誤消息
5、多個異常的捕獲
可以為一段腳本使用多個異常,來檢測多種情況。
可以使用多個 if..else 代碼塊,或一個switch 代碼塊,或者嵌套多個異常。這些異常能夠使用不同的 exception 類,并返回不同的錯誤消息:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
class customException extends Exception{ public function errorMessage(){ return = 'Error on line ' . $this ->getLine(). ' in ' . $this ->getFile(). ': <b>' . $this ->getMessage(). '</b> is not a valid E-Mail address' ; } } try { if (filter_var( $email , FILTER_VALIDATE_EMAIL) === FALSE){ throw new customException( $email ); } //check for "example" in mail address if ( strpos ( $email , "example" ) !== FALSE){ throw new Exception( "$email is an example e-mail" ); } } catch (customException $e ){ echo $e ->errorMessage(); } catch (Exception $e ){ echo $e ->getMessage(); } |
上面的代碼測試了兩種條件,如何任何條件不成立,則拋出一個異常:
執行 "try" 代碼塊,在第一個條件下,不會拋出異常。由于 e-mail 含有字符串 "example",第二個條件會觸發異常。"catch" 代碼塊會捕獲異常,并顯示恰當的錯誤消息,如果沒有捕獲 customException,只捕獲了 base exception,則在那里處理異常。
6、重新拋出異常
有時,當異常被拋出時,也許希望以不同于標準的方式對它進行處理。可以在一個 “catch” 代碼塊中再次拋出異常。腳本應該對用戶隱藏系統錯誤。對程序員來說,系統錯誤也許很重要,但是用戶對它們并不感興趣。為了讓用戶更容易使用,可以再次拋出帶有對用戶比較友好的消息的異常:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
class customException extends Exception{ public function errorMessage(){ return $this ->getMessage(). ' is not a valid E-Mail address.' ; } } try { try { if ( strpos ( $email , "example" ) !== FALSE){ throw new Exception( $email ); } } catch (Exception $e ){ //re-throw exception throw new customException( $email ); } } catch (customException $e ){ //display custom message echo $e ->errorMessage(); } |
上面的代碼檢測在郵件地址中是否含有字符串 "example"。如果有,則再次拋出異常:
把 $email 變量設置為一個有效的郵件地址,但含有字符串 "example"。"try" 代碼塊包含另一個 "try" 代碼塊,這樣就可以再次拋出異常。由于 e-mail 包含字符串 "example",因此觸發異常。"catch" 捕獲到該異常,并重新拋出 "customException"。捕獲到 "customException",并顯示一條錯誤消息。
如果在其目前的 "try" 代碼塊中異常沒有被捕獲,則它將在更高層級上查找 catch 代碼塊。
7、設置頂層異常處理器
set_exception_handler() 函數可設置處理所有未捕獲異常的用戶定義函數
1
2
3
4
5
|
function myException( $exception ){ echo "<b>Exception:</b> " , $exception ->getMessage(); } set_exception_handler( 'myException' ); throw new Exception( 'Uncaught Exception occurred' ); |
以上代碼的輸出應該類似這樣:
Exception: Uncaught Exception occurred
在上面的代碼中,不存在 "catch" 代碼塊,而是觸發頂層的異常處理程序。應該使用此函數來捕獲所有未被捕獲的異常。
但是對于系統自動拋出的錯誤,會先經過set_error_handler處理并拋出異常才能被set_exception_handler處理
1
2
3
4
5
6
7
8
|
function myException( $exception ){ echo "<b>Exception:</b> " , $exception ->getMessage(); } set_exception_handler( 'myException' ); set_error_handler( function (){ throw new Exception( '錯誤!' ); }); echo 10/0; //觸發被除數不能為0的警告 |
代碼執行結果:Exception: 錯誤!
總結
到此這篇關于PHP中異常處理機制的文章就介紹到這了,更多相關PHP異常處理機制內容請搜索服務器之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持服務器之家!
原文鏈接:https://juejin.im/post/6893305918243471373