PHP中的異常及其處理機(jī)制
上回文章中我們講到了錯(cuò)誤是編譯和語(yǔ)法運(yùn)行時(shí)會(huì)出現(xiàn)的,它們與邏輯無(wú)關(guān),是程序員在碼代碼時(shí)不應(yīng)該出現(xiàn)的,也就是說(shuō),這些錯(cuò)誤應(yīng)該是盡量避免帶到線上環(huán)境的,他們不能通過(guò)try...catch捕獲到。而異常則正好相反。
什么是異常?異常,指的是程序運(yùn)行中出現(xiàn)的不符合預(yù)期的情況,通常允許它發(fā)生,并交由相應(yīng)的異常處理來(lái)進(jìn)行處理。當(dāng)然,你也可以選擇忽略掉異常的處理,但是就像嚴(yán)重錯(cuò)誤一樣,代碼馬上會(huì)終止運(yùn)行。異常屬于業(yè)務(wù)邏輯上的錯(cuò)誤,基本上是我們?nèi)藶榈摹?/p>
還是先通過(guò)一個(gè)簡(jiǎn)單的代碼看下異常的拋出和捕獲:
function test(){ throw new Exception(’This is test Error...’);}try { test();} catch (Exception $e) { print_r($e);}
我們通過(guò) throw 來(lái)拋出異常,然后在調(diào)用方法時(shí)將方法包裹在 try...catch 塊中來(lái)捕獲拋出的異常。這就是異常最基礎(chǔ)的結(jié)構(gòu)。
從這里我們可以看出,異常基本都是通過(guò)我們手動(dòng)進(jìn)行拋出的,讓外部來(lái)進(jìn)行處理。在PHP內(nèi)部多數(shù)也是在類中會(huì)進(jìn)行異常的拋出,這就是面向?qū)ο蟮腻e(cuò)誤處理思想了。比如說(shuō)PDO類:
try { // $pdo = new PDO(); // Fatal error: Uncaught ArgumentCountError: PDO::__construct() expects at least 1 parameter, 0 given $pdo = new PDO(’’);} catch (PDOException $e) { print_r($e); // invalid data source name}
注意上面那行注釋的代碼,沒(méi)有傳參數(shù)是錯(cuò)誤,是無(wú)法捕獲的。而傳了的參數(shù)不對(duì),就是異常了,在PDO類的源碼中發(fā)現(xiàn)參數(shù)不對(duì)進(jìn)行了拋出。交給上層代碼也就是我們這些調(diào)用方來(lái)進(jìn)行捕獲。
接下來(lái),我們看下自定義的異常類和finally語(yǔ)句塊的使用。
自定義的異常類都會(huì)去繼承 Exception 類,這個(gè)類可以看做是所有異常的基類。它的結(jié)構(gòu)如下:
class Exception{ protected $message = ’Unknown exception’; // 異常信息 private $string; // __toString cache protected $code = 0;// 用戶自定義異常代碼 protected $file; // 發(fā)生異常的文件名 protected $line; // 發(fā)生異常的代碼行號(hào) private $trace; // backtrace private $previous;// previous exception if nested exception public function __construct($message = null, $code = 0, Exception $previous = null); final private function __clone(); // 不能被復(fù)制,如果clone異常類將直接產(chǎn)生致命錯(cuò)誤 final public function getMessage();// 返回異常信息 final public function getCode(); // 返回異常代碼 final public function getFile(); // 返回發(fā)生異常的文件名 final public function getLine(); // 返回發(fā)生異常的代碼行號(hào) final public function getTrace(); // backtrace() 數(shù)組 final public function getPrevious(); // 之前的 exception final public function getTraceAsString(); // 已格成化成字符串的 getTrace() 信息 // Overrideable public function __toString(); // 可輸出的字符串}
注意上面那行注釋的代碼,沒(méi)有傳參數(shù)是錯(cuò)誤,是無(wú)法捕獲的。而傳了的參數(shù)不對(duì),就是異常了,在PDO類的源碼中發(fā)現(xiàn)參數(shù)不對(duì)進(jìn)行了拋出。交給上層代碼也就是我們這些調(diào)用方來(lái)進(jìn)行捕獲。
接下來(lái),我們看下自定義的異常類和finally語(yǔ)句塊的使用。
自定義的異常類都會(huì)去繼承 Exception 類,這個(gè)類可以看做是所有異常的基類。它的結(jié)構(gòu)如下:
class Exception{ protected $message = ’Unknown exception’; // 異常信息 private $string; // __toString cache protected $code = 0;// 用戶自定義異常代碼 protected $file; // 發(fā)生異常的文件名 protected $line; // 發(fā)生異常的代碼行號(hào) private $trace; // backtrace private $previous;// previous exception if nested exception public function __construct($message = null, $code = 0, Exception $previous = null); final private function __clone(); // 不能被復(fù)制,如果clone異常類將直接產(chǎn)生致命錯(cuò)誤 final public function getMessage();// 返回異常信息 final public function getCode(); // 返回異常代碼 final public function getFile(); // 返回發(fā)生異常的文件名 final public function getLine(); // 返回發(fā)生異常的代碼行號(hào) final public function getTrace(); // backtrace() 數(shù)組 final public function getPrevious(); // 之前的 exception final public function getTraceAsString(); // 已格成化成字符串的 getTrace() 信息 // Overrideable public function __toString(); // 可輸出的字符串}
通過(guò)上述類定義,我們可以看出,我們能重寫 構(gòu)造函數(shù) 和 __toString() 方法,也能使用一些受保護(hù)的屬性。那么我們就來(lái)定義一個(gè)自定義的異常類吧。
class TestException extends Exception{ protected $code = 200; public function __construct($message = null, $code = 0, Exception $previous = null){$this->message = ’TestException:’ . $message; } public function __toString(){return ’code: ’ . $this->code . ’; ’ . $this->message; }}function test2(){ throw new TestException(’This is test2 Error...’);}try { test2();} catch (TestException $e) { echo $e, PHP_EOL; // code: 200; TestException:This is test2 Error...}
還是非常好理解的吧,大部分的PHP框架都會(huì)有自定義異常的組件或者能力供我們使用,因?yàn)楝F(xiàn)代框架還是以面向?qū)ο鬄榛A(chǔ)的,所以異常會(huì)定義的比較詳細(xì)。不同組件會(huì)提供不同的異常類來(lái)進(jìn)行異常的提示封裝。
接下來(lái)就是 finally 關(guān)鍵字,其實(shí)這個(gè)并沒(méi)有什么可多說(shuō)的,finally 的特點(diǎn)就是不管有沒(méi)有出現(xiàn)異常,都會(huì)去執(zhí)行 finally 關(guān)鍵字所定義代碼塊內(nèi)部的內(nèi)容。
try { test2();} catch (TestException $e) { echo $e, PHP_EOL; } finally { echo ’continue this code ...’, PHP_EOL;}// code: 200; TestException:This is test2 Error...// continue this code ...
說(shuō)了這么多,最后我們來(lái)結(jié)合上述內(nèi)容來(lái)處理下除0錯(cuò)誤的異常拋出。在文章開(kāi)頭已經(jīng)說(shuō)過(guò),錯(cuò)誤是應(yīng)該避免的,而異常是屬于邏輯業(yè)務(wù)的。所以當(dāng)我們接到一個(gè)需要做除法的參數(shù)時(shí),可以先判斷這個(gè)數(shù)是否為0,如果是0的話,就拋出異常讓上層調(diào)用者來(lái)處理,如果不是0的話,就讓它正常進(jìn)行除法運(yùn)算就好了。
function test3($d){ if ($d == 0) {throw new Exception(’除數(shù)不能為0’); } return 1 / $d;}try { echo test3(2), PHP_EOL;} catch (Exception $e) { echo ’Excepition:’ . $e->getMessage(), PHP_EOL;} finally { echo ’finally:繼續(xù)執(zhí)行!’, PHP_EOL;}// 0.5// finally:繼續(xù)執(zhí)行!try { echo test3(0), PHP_EOL;} catch (Exception $e) { echo ’Excepition:’ . $e->getMessage(), PHP_EOL;} finally { echo ’finally:繼續(xù)執(zhí)行!’, PHP_EOL;}// Excepition:除數(shù)不能為0// finally:繼續(xù)執(zhí)行!總結(jié)
異常相關(guān)的使用就是這些了,通過(guò)這兩篇文章,相信大家已經(jīng)對(duì)PHP的錯(cuò)誤和異常有了一些直觀的了解了。接下來(lái)的文章我們將一起對(duì)比下錯(cuò)誤和異常,并且說(shuō)明一下PHP7對(duì)錯(cuò)誤有了哪些改進(jìn)。內(nèi)容依然精彩,值得期待哦!!
以上就是PHP中的異常及其處理機(jī)制的詳細(xì)內(nèi)容,更多關(guān)于PHP 異常的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. el-input無(wú)法輸入的問(wèn)題和表單驗(yàn)證失敗問(wèn)題解決2. XML入門的常見(jiàn)問(wèn)題(三)3. XML 增、刪、改和查示例4. React實(shí)現(xiàn)一個(gè)倒計(jì)時(shí)hook組件實(shí)戰(zhàn)示例5. JavaScript中顏色模型的基礎(chǔ)知識(shí)與應(yīng)用詳解6. JavaScript快速實(shí)現(xiàn)一個(gè)顏色選擇器7. 報(bào)錯(cuò):XML頁(yè)無(wú)法顯示,下列標(biāo)記沒(méi)有被關(guān)閉解決方法8. CSS3實(shí)例分享之多重背景的實(shí)現(xiàn)(Multiple backgrounds)9. 前端html+css實(shí)現(xiàn)動(dòng)態(tài)生日快樂(lè)代碼10. 不要在HTML中濫用div
