請一定要注意型诚,沒有特殊說明:本例 ****PHP Version < 7
說起PHP異常處理,大家首先會想到try-catch娩嚼,那好爹橱,我們先看一段程序吧:有一個test.php文件,有一段簡單的PHP程序纵搁,內容如下吃衅,然后命令行執(zhí)行:php test.php
<?php
num;
6 } catch (Exception e->getMessage();
}
?>
我的問題是:這段程序能正確的捕捉到除0的錯誤信息嗎?
如果你回答能腾誉,那你就把這篇文章看完吧徘层!應該能學點東西峻呕。
本文章分5個部分介紹我的異常處理的理解:
一、異常與錯誤的概述
PHP中什么是異常:
程序在運行中出現(xiàn)不符合預期的情況干花,允許發(fā)生(你也不想讓他出現(xiàn)不正常的情況)但他是一種不正常的情況妄帘,按照我們的正常邏輯本不該出的錯誤,但仍然會出現(xiàn)的錯誤池凄,屬于邏輯和業(yè)務流程的錯誤抡驼,而不是編譯或者語法上的錯誤。
PHP中什么是錯誤:
屬于php腳本自身的問題肿仑,大部分情況是由錯誤的語法致盟,服務器環(huán)境導致,使得編譯器無法通過檢查尤慰,甚至無法運行的情況馏锡。warning、notice都是錯誤伟端,只是他們的級別不同而已杯道,并且錯誤是不能被try-catch捕獲的。
上面的說法是有前提條件的:
在PHP中责蝠,因為在其他語言中就不能這樣下結論了党巾,也就是說異常和錯誤的說法在不同的語言有不同的說法。在PHP中任何自身的錯誤或者是非正常的代碼都會當做錯誤對待霜医,并不會以異常的形式拋出齿拂,但是也有一些情況會當做異常和錯誤同時拋出(據(jù)說是,我沒有找到合適的例子)肴敛。也就是說署海,你想在數(shù)據(jù)庫連接失敗的時候自動捕獲異常是行不通的,因為這就不是異常值朋,是錯誤叹侄。但是在java中就不一樣了,他會把很多和預期不一致的行為當做異常來進行捕獲昨登。
PHP異常處理很雞肋趾代?
在上面的分析中我們可以看出,PHP并不能主動的拋出異常丰辣,但是你可以手動拋出異常撒强,這就很無語了禽捆,如果你知道哪里會出問題,你添加if else解決不就行了嗎飘哨,為啥還要手動拋出異常胚想,既然能手動拋出就證明這個不是異常,而是意料之中芽隆。以我的理解浊服,這就是PHP異常處理雞肋的地方(不一定對啊)胚吁。所以PHP的異常機制不是那么的完美牙躺,但是使用過框架的同學都知道有這個情況:你在框架中直接寫開頭那段php“自動”捕獲異常的代碼是可以的,這是為什么腕扶?看過源碼的同學都知道框架中都會涉及三個函數(shù):register_shutdown_function孽拷,set_error_handler,set_exception_handler后面我會重點講解著三個黑科技半抱,通過這幾個函數(shù)我們可以實現(xiàn)PHP假自動捕獲異常和錯誤脓恕。
二、ERROR的級別
只有熟悉錯誤級別才能對錯誤捕捉有更好的認識窿侈。 ERROR有不同的錯誤級別炼幔,我之前的一篇文章中有寫到:http://www.cnblogs.com/zyf-zhaoyafei/p/3649434.html
下面我再總結性的給出這幾類錯誤級別:
](javascript:void(0); "復制代碼")
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;"> 1 Fatal Error:致命錯誤(腳本終止運行)
2 E_ERROR // 致命的運行錯誤,錯誤無法恢復棉磨,暫停執(zhí)行腳本
3 E_CORE_ERROR // PHP啟動時初始化過程中的致命錯誤
4 E_COMPILE_ERROR // 編譯時致命性錯江掩,就像由Zend腳本引擎生成了一個E_ERROR
5 E_USER_ERROR // 自定義錯誤消息学辱。像用PHP函數(shù)trigger_error(錯誤類型設置為:E_USER_ERROR)
6
7 Parse Error:編譯時解析錯誤乘瓤,語法錯誤(腳本終止運行)
8 E_PARSE //編譯時的語法解析錯誤
9
10 Warning Error:警告錯誤(僅給出提示信息,腳本不終止運行) 11 E_WARNING // 運行時警告 (非致命錯誤)策泣。
12 E_CORE_WARNING // PHP初始化啟動過程中發(fā)生的警告 (非致命錯誤) 衙傀。
13 E_COMPILE_WARNING // 編譯警告
14 E_USER_WARNING // 用戶產生的警告信息
15
16 Notice Error:通知錯誤(僅給出通知信息,腳本不終止運行) 17 E_NOTICE // 運行時通知萨咕。表示腳本遇到可能會表現(xiàn)為錯誤的情況.
18 E_USER_NOTICE // 用戶產生的通知信息统抬。</pre>
](javascript:void(0); "復制代碼")
由此可知有5類是產生ERROR級別的錯誤,這種錯誤直接導致PHP程序退出危队。
可以定義成:
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;">1 ERROR = E_ERROR | E_CORE_ERROR | E_COMPILE_ERROR | E_USER_ERROR | E_PARSE</pre>
三聪建、PHP異常處理中的黑科技
前面提到框架中是可以捕獲所有的錯誤和異常的,之所以能實現(xiàn)應該是使用了黑科技茫陆,哈哈金麸!其實也不是什么黑科技,主要是三個重要的函數(shù):
1:set_error_handler()
看到這個名字估計就知道什么意思了簿盅,這個函數(shù)用于捕獲錯誤挥下,設置一個用戶自定義的錯誤處理函數(shù)揍魂。
](javascript:void(0); "復制代碼")
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;">1 <?php 2 set_error_handler('zyferror'); 3 function zyferror(message,
line) 4 { 5 var_dump('<b>set_error_handler: ' .
message . ' in ' .
line . ' line .</b><br />'); 6 } 7 ?></pre>
](javascript:void(0); "復制代碼")
當程序出現(xiàn)錯誤的時候自動調用此方法,不過需要注意一下兩點:第一棚瘟,如果存在該方法现斋,相應的error_reporting()就不能在使用了。所有的錯誤都會交給自定義的函數(shù)處理偎蘸。第二庄蹋,此方法不能處理以下級別的錯誤:E_ERROR、 E_PARSE迷雪、 E_CORE_ERROR蔓肯、 E_CORE_WARNING、 E_COMPILE_ERROR振乏、 E_COMPILE_WARNING蔗包,set_error_handler() 函數(shù)所在文件中產生的E_STRICT,該函數(shù)只能捕獲系統(tǒng)產生的一些Warning慧邮、Notice級別的錯誤调限。
并且他有多種調用的方法:
](javascript:void(0); "復制代碼")
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;">1 <?php 2 // 直接傳函數(shù)名 NonClassFunction
3 set_error_handler('function_name'); 4
5 // 傳 class_name && function_name
6 set_error_handler(array('class_name', 'function_name')); 7 ?></pre>
](javascript:void(0); "復制代碼")
2:register_shutdown_function()
捕獲PHP的錯誤:Fatal Error、Parse Error等误澳,這個方法是PHP腳本執(zhí)行結束前最后一個調用的函數(shù)耻矮,比如腳本錯誤、die()忆谓、exit裆装、異常、正常結束都會調用倡缠,多么牛逼的一個函數(shù)吧诿狻!通過這個函數(shù)就可以在腳本結束前判斷這次執(zhí)行是否有錯誤產生昙沦,這時就要借助于一個函數(shù):error_get_last()琢唾;這個函數(shù)可以拿到本次執(zhí)行產生的所有錯誤。error_get_last();返回的信息:
[type] - 錯誤類型
[message] - 錯誤消息
[file] - 發(fā)生錯誤所在的文件
[line] - 發(fā)生錯誤所在的行
](javascript:void(0); "復制代碼")
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;">1 <?php 2 register_shutdown_function('zyfshutdownfunc'); 3 function zyfshutdownfunc() 4 { 5 if (error['type'] . ' Msg: ' .
error['file'] . ' on line ' . $error['line'] . '</b>'); 7 } 8 } 9 ?></pre>
](javascript:void(0); "復制代碼")
通過這種方法就可以巧妙的打印出程序結束前所有的錯誤信息盾饮。但是我在測試的時候我發(fā)現(xiàn)并不是所有的錯誤終止后都會調用這個函數(shù)采桃,可以看下面的一個測試文件,內容是:
[](javascript:void(0); "復制代碼")
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;"> 1 <?php
2 register_shutdown_function('zyfshutdownfunc');
3 function zyfshutdownfunc() 4 {
5 if (error['type'] . ' Msg: ' .
error['file'] . ' on line ' . $error['line'] . '</b>');
7 }
8 }
9 var_dump(23+-+); //此處語法錯誤
10 ?></pre>
](javascript:void(0); "復制代碼")
自己可以試一下丘损,你可以看到根本就不會觸發(fā)zyfshutdownfunc()函數(shù)普办,其實這是一個語法錯誤,直接報了一個:
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;">1 <?php 2 Parse error: syntax error, unexpected ')' in /www/mytest/exception/try-catch.php on line 71
3 ?></pre>
由此引出一個奇葩的問題:問什么不能觸發(fā)徘钥,為什么框架中是可以的衔蹲?其實原因很簡單,只在parse-time出錯時是不會調用本函數(shù)的吏饿。只有在run-time出錯的時候踪危,才會調用本函數(shù)蔬浙,我的理解是語法檢查器前沒有執(zhí)行register_shutdown_function()去把需要注冊的函數(shù)放到調用的堆棧中,所以就根本不會運行贞远。那框架中為什么任何錯誤都能進入到register_shutdown_function()中呢畴博,其實在框架中一般會有統(tǒng)一的入口index.php,然后每個類庫文件都會通過include ** 的方式加載到index.php中蓝仲,相當與所有的程序都會在index.php中聚集俱病,同樣,你寫的具有語法錯誤的文件也會被引入到入口文件中袱结,這樣的話亮隙,調用框架,執(zhí)行index.php垢夹,index.php本身并沒有語法錯誤溢吻,也就不會產生parse-time錯誤,而是 include 文件出錯了果元,是run-time的時候出錯了促王,所以框架執(zhí)行完之后就會觸發(fā)register_shutdown_function();
所以現(xiàn)在可是試一下這個寫法,這樣就會觸發(fā)zyfshutdownfunc()回調了:
](javascript:void(0); "復制代碼")
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;"> 1 a.php文件
2 <?php
3 // 模擬語法錯誤
4 var_dump(23+-+);
5 ?>
6
7 b.php文件
8 <?php
9 register_shutdown_function('zyfshutdownfunc'); 10 function zyfshutdownfunc() 11 { 12 if (error['type'] . ' Msg: ' .
error['file'] . ' on line ' . $error['line'] . '</b>'); 14 } 15 } 16 require 'a.php'; 17 ?></pre>
](javascript:void(0); "復制代碼")
3:set_exception_handler()
設置默認的異常處理程序而晒,用在沒有用try/catch塊來捕獲的異常蝇狼,也就是說不管你拋出的異常有沒有人捕獲,如果沒有人捕獲就會進入到該方法中倡怎,并且在回調函數(shù)調用后異常會中止迅耘。看一下用法:
](javascript:void(0); "復制代碼")
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;">1 <?php 2 set_exception_handler('zyfexception'); 3 function zyfexception(exception->getMessage() . '</b>'); 6 } 7 throw new Exception("zyf exception"); 8 ?></pre>
](javascript:void(0); "復制代碼")
四监署、巧妙的捕獲錯誤和異常
1:把錯誤以異常的形式拋出(不能完全拋出)
由上面的講解我們知道颤专,php中的錯誤是不能以異常的像是捕獲的,但是我們需要讓他們拋出焦匈,已達到擴展 try-catch的影響范圍血公,我們前面講到過set_error_handler() 方法,他是干嘛用的缓熟,他是捕獲錯誤的,所以我們就可以借助他來吧錯誤捕獲摔笤,然后再以異常的形式拋出够滑,ok,試試下面的寫法:
](javascript:void(0); "復制代碼")
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;"> 1 <?php
2 set_error_handler('zyferror');
3 function zyferror(message,
line)
4 {
5 throw new \Exception(num = 0;
9 try { 10 echo 1/e){ 13 echo $e->getMessage(); 14 } 15 ?></pre>
](javascript:void(0); "復制代碼")
好了吕世,試一下彰触,會打印出:
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;">1 Division by zero zyf123</pre>
流程:本來是除0錯誤,然后觸發(fā)set_error_handler()命辖,在set_error_handler()中相當與殺了個回馬槍况毅,再把錯誤信息以異常的形式拋出來分蓖,這樣就可以實現(xiàn)錯誤以異常的形式拋出。大家要注意:這樣做是有缺點的尔许,會受到set_error_handler()函數(shù)捕獲級別的限制么鹤。
**2:捕獲所有的錯誤**
由set_error_handler()可知,他能夠捕獲一部分錯誤味廊,不能捕獲系統(tǒng)級E_ERROR蒸甜、E_PARSE等錯誤,但是這部分可以由register_shutdown_function()捕獲余佛。所以兩者結合能出現(xiàn)很好的功能柠新。
看下面的程序:
[](javascript:void(0); "復制代碼")
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;"> 1 a.php內容:
2 <?
3 // 模擬Fatal error錯誤 4 //test();
5
6 // 模擬用戶產生ERROR錯誤
7 //trigger_error('zyf-error', E_USER_ERROR);
8
9 // 模擬語法錯誤
10 var_dump(23+-+); 11
12 // 模擬Notice錯誤 13 //echo $f; 14
15 // 模擬Warning錯誤 16 //echo '123'; 17 //ob_flush(); 18 //flush(); 19 //header("Content-type:text/html;charset=gb2312");
20 ?>
21 b.php內容: 22 <?
23 error_reporting(0); 24 register_shutdown_function('zyfshutdownfunc'); 25 function zyfshutdownfunc() 26 { 27 if (error['type'] . ' Msg: ' .
error['file'] . ' on line ' .
type,
file,
type . ':' .
file . ' on ' . $line . ' line .</b><br />'); 36 } 37
38 require 'a.php'; 39 ?></pre>
](javascript:void(0); "復制代碼")
到此就可以解釋開頭的那個程序了吧,test.php 如果是單文件執(zhí)行是不能捕獲到錯誤的辉巡,如果你在框架中執(zhí)行就是可以的恨憎,當然你按照我上面介紹的來擴展也是可以的。
五郊楣、自定義異常處理和異常嵌套
1:自定義異常處理
在復雜的系統(tǒng)中框咙,我們往往需要自己捕獲我們需要特殊處理的異常,這些異沉「剩可能是特殊情況下拋出的喇嘱。所以我們就自己定義一個異常捕獲類,該類必須是 exception 類的一個擴展塞栅,該類繼承了 PHP 的 exception 類的所有屬性者铜,并且我們可以添加自定義的函數(shù),使用的時候其實和之前的一樣放椰,大致寫法如下:
[](javascript:void(0); "復制代碼")
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;"> 1 <?php
2 class zyfException extends Exception
3 {
4 public function errorzyfMessage() 5 {
6 return 'Error line ' . this->getFile()
7 .': <b>' . age = 10; 12 try { 13
age); 14 if(
age); 16 } 17
18 } catch (zyfException e->errorzyfMessage(); 20
21 } 22 ?></pre>
](javascript:void(0); "復制代碼")
2:異常嵌套
異常嵌套是比較常見的寫法作烟,在自定義的異常處理中,try 塊中可以定義多個異常捕獲砾医,然后分層傳遞異常拿撩,理解和冒泡差不多,看下面的實現(xiàn):
[](javascript:void(0); "復制代碼")
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;"> 1 <?php
2 age = intval(
age > 60) {
6 throw new zyfException(age <= 0) { 10 throw new Exception(
e) { 14 echo
e) { 17 echo $e->getMessage(); 18 } 19 ?></pre>
](javascript:void(0); "復制代碼")
當然也可以在catch中再拋出異常給上層:
[](javascript:void(0); "復制代碼")
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;"> 1 <?php
2 age = intval(
age > 60) {
7 throw new Exception(e) { 11 throw new zyfException(
e) { 16 echo $e->errorzyfMessage(); 17 } 18 ?></pre>
](javascript:void(0); "復制代碼")
六如蚜、PHP7中的異常處理
現(xiàn)在寫PHP必須考慮版本情況压恒,上面的寫法在PHP7中大部分都能實現(xiàn),但是也會有不同點错邦,在PHP7更新中有一條:更多的Error變?yōu)榭刹东@的Exception探赫,現(xiàn)在的PHP7實現(xiàn)了一個全局的throwable接口,原來老的Exception和其中一部分Error實現(xiàn)了這個接口(interface)撬呢,PHP7中更多的Error變?yōu)榭刹东@的Exception返回給捕捉器伦吠,這樣其實和前面提到的擴展try-catch影響范圍一樣,但是如果不捕獲則還是按照Error對待,看下面兩個:
](javascript:void(0); "復制代碼")
<pre style="margin: 0px; padding: 0px; white-space: pre-wrap; word-wrap: break-word; font-family: "Courier New" !important; font-size: 12px !important;"> 1 <?php
2 try { 3 test();
4
5 } catch(Throwable e->getMessage() . ' zyf';
7 }
8
9 try { 10 test(); 11
12 } catch(Error e->getMessage() . ' zyf'; 14 } 15 ?></pre>
](javascript:void(0); "復制代碼")
因為PHP7實現(xiàn)了throwable接口毛仪,那么就可以使用第一個這種方式來捕獲異常搁嗓。又因為部分Error實現(xiàn)了接口,并且更多的Error變?yōu)榭刹东@的Exception箱靴,那么就可以使用第二種方式來捕獲異常腺逛。下面是在網上找的PHP7的異常層次樹:
Throwable
Exception 異常
...
Error 錯誤
ArithmeticError 算數(shù)錯誤
DivisionByZeroError 除數(shù)為0的錯誤
AssertionError 聲明錯誤
ParseError 解析錯誤
TypeError 類型錯誤
就寫到這吧,寫得手疼刨晴,關于錯誤和異常處理的大致就寫這么多屉来,有什么錯誤請在評論中給出,多謝大家狈癞。
注意:
1茄靠、本博客同步更新到我的個人網站:http://www.zhaoyafei.cn
2、本文屬原創(chuàng)內容蝶桶,為了尊重他人勞動慨绳,轉載請注明本文地址: