錯誤與異常 錯誤,可以理解程式本身的錯誤,例如語法錯誤。而異常則更偏向於程式運行不符合預期或者不符合正常流程;對於 PHP 語言而言,處理錯誤和處理異常使用的機制完全不同,因此很容易讓人產生困惑。 例如,我們希望通過捕獲異常來處理除數為 0 的情況,但是在捕獲到異常之前,PHP 就觸發了錯誤。 1 ...
錯誤與異常
錯誤,可以理解程式本身的錯誤,例如語法錯誤。而異常則更偏向於程式運行不符合預期或者不符合正常流程;對於 PHP 語言而言,處理錯誤和處理異常使用的機制完全不同,因此很容易讓人產生困惑。
例如,我們希望通過捕獲異常來處理除數為 0 的情況,但是在捕獲到異常之前,PHP 就觸發了錯誤。
1 try { 2 $a = 5 / 0; 3 } catch (Exception $e) { 4 $e->getMessage(); 5 $a = -1; // 通過異常來處理 $a 為 0 的情況,但是實際上,捕獲不到該異常 6 } 7 8 echo $a; 9 // PHP Warning: Division by zero
也就是說,PHP 將除數為 0 的情況當成了錯誤而觸發,而不會自動拋出異常,因此沒法捕獲。類似的,在很多情況下,PHP 都沒辦法自動拋出異常。只能通過 if - else
語句判斷再結合 throw
方法來並手動拋出異常。
上述情況的發生,主要還是因為異常機制是 PHP 向面向對象演進後得到的產物。而在此之前 PHP 的報錯主要還是通過錯誤機制,因此,在很多情況下,PHP 的錯誤要比異常更有價值。不過 PHP7 開始統一這兩者,使錯誤也可以像異常那樣拋出(這部分內容將放在異常部分講解)。
錯誤級別
PHP 中的錯誤可理解為 使腳本不運行不正常的情況,根據錯誤級別從高到低可劃分為五類
- Parse error 或 Syntax Error - 語法解析錯誤,觸發該錯誤後,腳本完全無法運行;
- Fatal Error - 致命錯誤,觸發該錯誤後,後面的腳本無法繼續執行;
- Warning Error - 出現比較不恰當的地方,腳本可繼續執行;
- Notice Error - 出現不恰當的地方,但是程度比 Warning Error 低,腳本可繼續執行;
- Deprecated Error - 不推薦這麼使用,未來可能會廢棄,腳本可繼續執行;
預設情況下,PHP 觸發錯誤,並顯示錯誤的級別及對應的提示。
Parse Error 示例 - 語句結尾不寫分號
1 echo "abc" 2 // PHP Parse error: syntax error, unexpected end of file, expecting ',' or ';
Fatal Error 示例 - 使用不存在的函數
1 echo "before\n"; 2 foo(); 3 echo "after"; // 本行無法繼續執行 4 // before 5 // PHP Fatal error: Uncaught Error: Call to undefined function foo()
Warning Error 示例 - 引入不存在的文件
1 $a = "foo"; 2 include('bar.php'); 3 echo $a; // 程式繼續執行 4 // PHP Warning: include(bar.php): failed to open stream: No such file or directory ... 5 // foo 6 Notice Error 示例 - 輸出不存在的變數 7 8 echo $foo; 9 echo 12345; 10 // PHP Notice: Undefined variable: foo 11 // 12345
Deprecated Error 示例 - 在一些字元串函數中傳入數字而非字元串
1 strpos('12345', 3); 2 // PHP Deprecated: strpos(): Non-string needles will be interpreted as strings in the future
除了預設觸發消息外,用戶也可以使用 set_error_handler
函數自定義錯誤處理,大多數錯誤類型都可以進行自定義處理,除了 E_ERROR
、 E_PARSE
、 E_CORE_ERROR
、 E_CORE_WARNING
、 E_COMPILE_ERROR
、 E_COMPILE_WARNING
外。
1 set_error_handler ( callable $error_handler [, int $error_types = E_ALL | E_STRICT ] ) : mixed
示例
1 <?php 2 // E_ALL - 處理全部錯誤類型 3 set_error_handler('customError', E_ALL); 4 5 /** 6 * @param int $errno 錯誤的級別 7 * @param string $errstr 錯誤的信息 8 * @param string $errfile 錯誤的文件名(可選) 9 * @param string $errline 錯誤發生的行號(可選) 10 */ 11 function customError(int $errno, string $errstr, string $errfile, string $errline) 12 { 13 echo sprintf('錯誤消息為 %s', $errstr); 14 } 15 16 $a = 5 / 0; // 錯誤消息為 Division by zero
用戶也可以通過 trigger_error
函數來手動觸發一個用戶級別的錯誤(E_USER_ERROR
、E_USER_WARNING
、E_USER_NOTICE
、E_USER_DEPRECATED
)。
1 function division($a, $b) { 2 if($b == 0){ 3 @trigger_error("0 不能作為除數", E_USER_NOTICE); 4 return -1; 5 } 6 return $a / $b; 7 } 8 9 echo division(10, 0);
與錯誤有關的配置
一些錯誤處理相關的常用配置
error_reporting
- 設置錯誤的報告級別display_errors
- 是否顯示錯誤display_startup_error
- 是否顯示 PHP 啟動過程中的顯示log_errors
- 設置是否將腳本運行的錯誤信息記錄到伺服器錯誤日誌或者 error_log 之中
《Modern PHP》提出了四個規則
- 一定要讓 PHP 報告錯誤;
- 在開發環境中要顯示錯誤;
- 在生產環境中不能顯示錯誤;
- 在開發環境和生產環境中都要記錄錯誤;
開發環境推薦配置
1 display_errors = On 2 display_startup_error = On 3 error_reporting = -1 4 log_errors = On
生產環境推薦配置
1 display_errors = Off 2 display_startup_error = Off 3 ; 報告 Notice 以外的所有錯誤 4 error_reporting = E_ALL & ~E_NOTICE 5 log_errors = On 6
Symfony 編碼規範相關
異常和錯誤消息字元串必須使用 sprintf
來進行拼接;
throw new CommandNotFoundException(sprintf('Command "%s" does not exist.', $name));
當錯誤類型為 E_USER_DEPRECATED
時,需要添加 @
@trigger_error("foo", E_USER_DEPRECATED);
參考資料
- PHP Errors: 4 Different Types (Warning, Parse, Fatal, and Notice Error)
- PHP: 預定義常量 - Manual
- PHP 核心技術與最佳實踐 (豆瓣)
- PHP: 運行時配置 - Manual
- PHP 規範 - Symfony 代碼規範 | PHP 技術論壇
- 很多PHPer在進階的時候總會遇到一些問題和瓶頸,業務代碼寫多了沒有方向感,不知道該從那裡入手去提升,對此我整理了一些資料,包括但不限於:分散式架構、高可擴展、高性能、高併發、伺服器性能調優、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、Mysql優化、shell腳本、Docker、微服務、Nginx等多個知識點高級進階乾貨需要的可以免費分享給大家,需要的加群(點擊→)677079770