PHP的异常处理异常捕获错误类
1.錯誤分類
Fatal Error:致命錯誤(腳本終止運行)E_ERROR // 致命的運行錯誤,錯誤無法恢復,暫停執行腳本E_CORE_ERROR // PHP啟動時初始化過程中的致命錯誤E_COMPILE_ERROR // 編譯時致命性錯,就像由Zend腳本引擎生成了一個E_ERRORE_USER_ERROR // 自定義錯誤消息。像用PHP函數trigger_error(錯誤類型設置為:E_USER_ERROR)Parse Error:編譯時解析錯誤,語法錯誤(腳本終止運行)E_PARSE //編譯時的語法解析錯誤Warning Error:警告錯誤(僅給出提示信息,腳本不終止運行)E_WARNING // 運行時警告 (非致命錯誤)。E_CORE_WARNING // PHP初始化啟動過程中發生的警告 (非致命錯誤) 。E_COMPILE_WARNING // 編譯警告E_USER_WARNING // 用戶產生的警告信息Notice Error:通知錯誤(僅給出通知信息,腳本不終止運行)E_NOTICE // 運行時通知。表示腳本遇到可能會表現為錯誤的情況.E_USER_NOTICE // 用戶產生的通知信息。2.set_error_handler():自定義錯誤處理函數
看到這個名字估計就知道什么意思了,這個函數用于捕獲錯誤,設置一個用戶自定義的錯誤處理函數。 <?phpset_error_handler('zyferror');function zyferror($type, $message, $file, $line){var_dump('<b>set_error_handler: ' . $type . ':' . $message . ' in ' . $file . ' on ' . $line . ' line .</b><br />');} ?>當程序出現錯誤的時候自動調用此方法,不過需要注意一下兩點:第一,如果存在該方法,相應的error_reporting()就不能在使用了。所有的錯誤都會交給自定義的函數處理。第二,此方法不能處理以下級別的錯誤:E_ERROR、 E_PARSE、 E_CORE_ERROR、 E_CORE_WARNING、 E_COMPILE_ERROR、 E_COMPILE_WARNING,set_error_handler() 函數所在文件中產生的E_STRICT,該函數只能捕獲系統產生的一些Warning、Notice級別的錯誤。
<?php// 直接傳函數名 NonClassFunctionset_error_handler('function_name'); // 傳 class_name && function_nameset_error_handler(array('class_name', 'function_name')); ?>3.register_shutdown_function():是腳本關閉時候最后一個執行的函數
捕獲PHP的錯誤:Fatal Error、Parse Error等,這個方法是PHP腳本執行結束前最后一個調用的函數,比如腳本錯誤、die()、exit、異常、正常結束都會調用,多么牛逼的一個函數啊!通過這個函數就可以在腳本結束前判斷這次執行是否有錯誤產生,這時就要借助于一個函數:error_get_last();這個函數可以拿到本次執行產生的所有錯誤。error_get_last();返回的信息:
[type]?????????? - 錯誤類型
[message] - 錯誤消息
[file]????????????? - 發生錯誤所在的文件
[line] ? ? ? ? ? ? - 發生錯誤所在的行
通過這種方法就可以巧妙的打印出程序結束前所有的錯誤信息。但是我在測試的時候我發現并不是所有的錯誤終止后都會調用這個函數,可以看下面的一個測試文件,內容是:
<?phpregister_shutdown_function('zyfshutdownfunc');function zyfshutdownfunc(){if ($error = error_get_last()) {var_dump('<b>register_shutdown_function: Type:' . $error['type'] . ' Msg: ' . $error['message'] . ' in ' . $error['file'] . ' on line ' . $error['line'] . '</b>');}}var_dump(23+-+); //此處語法錯誤 ?>自己可以試一下,你可以看到根本就不會觸發zyfshutdownfunc()函數,其實這是一個語法錯誤,直接報了一個:
1 <?php 2 Parse error: syntax error, unexpected ')' in /www/mytest/exception/try-catch.php on line 71 3 ?>由此引出一個奇葩的問題:問什么不能觸發,為什么框架中是可以的?其實原因很簡單,只在parse-time出錯時是不會調用本函數的。只有在run-time出錯的時候,才會調用本函數,我的理解是語法檢查器前沒有執行register_shutdown_function()去把需要注冊的函數放到調用的堆棧中,所以就根本不會運行。那框架中為什么任何錯誤都能進入到register_shutdown_function()中呢,其實在框架中一般會有統一的入口index.php,然后每個類庫文件都會通過include ** 的方式加載到index.php中,相當與所有的程序都會在index.php中聚集,同樣,你寫的具有語法錯誤的文件也會被引入到入口文件中,這樣的話,調用框架,執行index.php,index.php本身并沒有語法錯誤,也就不會產生parse-time錯誤,而是 include 文件出錯了,是run-time的時候出錯了,所以框架執行完之后就會觸發register_shutdown_function();
所以現在可是試一下這個寫法,這樣就會觸發zyfshutdownfunc()回調了:
4.set_exception_handler():設置異常處理函數
設置默認的異常處理程序,用在沒有用try/catch塊來捕獲的異常,也就是說不管你拋出的異常有沒有人捕獲, 如果沒有人捕獲就會進入到該方法中,并且在回調函數調用后異常會中止。看一下用法:<?phpset_exception_handler('zyfexception');function zyfexception($exception){var_dump("<b>set_exception_handler: Exception: " . $exception->getMessage() . '</b>');}throw new Exception("zyf exception"); ?>5.把錯誤以異常的形式拋出(不能完全拋出)
由上面的講解我們知道,php中的錯誤是不能以異常的像是捕獲的,但是我們需要讓他們拋出,已達到擴展 try-catch的影響范圍,我們前面講到過set_error_handler() 方法,他是干嘛用的,他是捕獲錯誤的,所以我們就可以借助他來吧錯誤捕獲,然后再以異常的形式拋出,ok,試試下面的寫法:
<?phpset_error_handler('zyferror');function zyferror($type, $message, $file, $line){throw new \Exception($message . 'zyf錯誤當做異常');}$num = 0;try {echo 1/$num;} catch (Exception $e){echo $e->getMessage();} ?> 好了,試一下,會打印出: Division by zerozyf錯誤當做異常流程:本來是除0錯誤,然后觸發set_error_handler(),在set_error_handler()中相當與殺了個回馬槍,再把錯誤信息以異常的形式拋出來,這樣就可以實現錯誤以異常的形式拋出。大家要注意:這樣做是有缺點的,會受到set_error_handler()函數捕獲級別的限制。
6.捕獲所有的錯誤
?由set_error_handler()可知,他能夠捕獲一部分錯誤,不能捕獲系統級E_ERROR、E_PARSE等錯誤,但是這部分可以由register_shutdown_function()捕獲。所以兩者結合能出現很好的功能。
??????? 看下面的程序:
到此就可以解釋開頭的那個程序了吧,test.php 如果是單文件執行是不能捕獲到錯誤的,如果你在框架中執行就是可以的,當然你按照我上面介紹的來擴展也是可以的。
7.自定義異常處理和異常嵌套
1:自定義異常處理
在復雜的系統中,我們往往需要自己捕獲我們需要特殊處理的異常,這些異常可能是特殊情況下拋出的。所以我們就自己定義一個異常捕獲類,該類必須是 exception 類的一個擴展,該類繼承了 PHP 的 exception 類的所有屬性,并且我們可以添加自定義的函數,使用的時候其實和之前的一樣,大致寫法如下:
<?phpclass zyfException extends Exception{public function errorzyfMessage(){return 'Error line ' . $this->getLine().' in ' . $this->getFile().': <b>' . $this->getMessage() . '</b> Must in (0 - 60)';}}$age = 10;try {$age = intval($age);if($age > 60) {throw new zyfException($age);}} catch (zyfException $e) {echo $e->errorzyfMessage();} ?>2:異常嵌套
異常嵌套是比較常見的寫法,在自定義的異常處理中,try 塊中可以定義多個異常捕獲,然后分層傳遞異常,理解和冒泡差不多,看下面的實現:
<?php$age = 10;try {$age = intval($age);if($age > 60) {throw new zyfException($age);}if ($age <= 0) {throw new Exception($age . ' must > 0');}} catch (zyfException $e) {echo $e->errorzyfMessage();} catch(Exception $e) {echo $e->getMessage();} ?> 當然也可以在catch中再拋出異常給上層: <?php$age = 100;try {try {$age = intval($age);if($age > 60) {throw new Exception($age);}} catch (Exception $e) {throw new zyfException($age);}} catch (zyfException $e) {echo $e->errorzyfMessage();} ?>8.PHP7中的異常處理
現在寫PHP必須考慮版本情況,上面的寫法在PHP7中大部分都能實現,但是也會有不同點,在PHP7更新中有一條:更多的Error變為可捕獲的Exception,現在的PHP7實現了一個全局的throwable接口,原來老的Exception和其中一部分Error實現了這個接口(interface),PHP7中更多的Error變為可捕獲的Exception返回給捕捉器,這樣其實和前面提到的擴展try-catch影響范圍一樣,但是如果不捕獲則還是按照Error對待,看下面兩個:
<?phptry {test();} catch(Throwable $e) {echo $e->getMessage() . ' zyf';}try {test();} catch(Error $e) {echo $e->getMessage() . ' zyf';} ?>?因為PHP7實現了throwable接口,那么就可以使用第一個這種方式來捕獲異常。又因為部分Error實現了接口,并且更多的Error變為可捕獲的Exception,那么就可以使用第二種方式來捕獲異常。下面是在網上找的PHP7的異常層次樹:
Throwable
Exception 異常
...
Error 錯誤
ArithmeticError 算數錯誤
DivisionByZeroError 除數為0的錯誤
AssertionError 聲明錯誤
ParseError 解析錯誤
TypeError 類型錯誤
這個很詳細?https://www.cnblogs.com/zyf-zhaoyafei/p/6928149.html
總結
以上是生活随笔為你收集整理的PHP的异常处理异常捕获错误类的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 电竞是体育项目么 国内电竞人才缺口
- 下一篇: Linux解决编译Apache出现的问题