PHP中的異常及其處理機(jī)制
上回文章中我們講到了錯(cuò)誤是編譯和語法運(yùn)行時(shí)會(huì)出現(xiàn)的,它們與邏輯無關(guān),是程序員在碼代碼時(shí)不應(yīng)該出現(xiàn)的,也就是說,這些錯(cuò)誤應(yīng)該是盡量避免帶到線上環(huán)境的,他們不能通過try...catch捕獲到。而異常則正好相反。
什么是異常?
異常,指的是程序運(yùn)行中出現(xiàn)的不符合預(yù)期的情況,通常允許它發(fā)生,并交由相應(yīng)的異常處理來進(jìn)行處理。當(dāng)然,你也可以選擇忽略掉異常的處理,但是就像嚴(yán)重錯(cuò)誤一樣,代碼馬上會(huì)終止運(yùn)行。異常屬于業(yè)務(wù)邏輯上的錯(cuò)誤,基本上是我們?nèi)藶榈摹?/p>
還是先通過一個(gè)簡(jiǎn)單的代碼看下異常的拋出和捕獲:
function test() { throw new Exception('This is test Error...'); } try { test(); } catch (Exception $e) { print_r($e); }
我們通過 throw 來拋出異常,然后在調(diào)用方法時(shí)將方法包裹在 try...catch 塊中來捕獲拋出的異常。這就是異常最基礎(chǔ)的結(jié)構(gòu)。
從這里我們可以看出,異?;径际峭ㄟ^我們手動(dòng)進(jìn)行拋出的,讓外部來進(jìn)行處理。在PHP內(nèi)部多數(shù)也是在類中會(huì)進(jìn)行異常的拋出,這就是面向?qū)ο蟮腻e(cuò)誤處理思想了。比如說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 }
注意上面那行注釋的代碼,沒有傳參數(shù)是錯(cuò)誤,是無法捕獲的。而傳了的參數(shù)不對(duì),就是異常了,在PDO類的源碼中發(fā)現(xiàn)參數(shù)不對(duì)進(jìn)行了拋出。交給上層代碼也就是我們這些調(diào)用方來進(jìn)行捕獲。
接下來,我們看下自定義的異常類和finally語句塊的使用。
自定義的異常類都會(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(); // 可輸出的字符串 }
注意上面那行注釋的代碼,沒有傳參數(shù)是錯(cuò)誤,是無法捕獲的。而傳了的參數(shù)不對(duì),就是異常了,在PDO類的源碼中發(fā)現(xiàn)參數(shù)不對(duì)進(jìn)行了拋出。交給上層代碼也就是我們這些調(diào)用方來進(jìn)行捕獲。
接下來,我們看下自定義的異常類和finally語句塊的使用。
自定義的異常類都會(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(); // 可輸出的字符串 }
通過上述類定義,我們可以看出,我們能重寫 構(gòu)造函數(shù) 和 __toString() 方法,也能使用一些受保護(hù)的屬性。那么我們就來定義一個(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ì)提供不同的異常類來進(jìn)行異常的提示封裝。
接下來就是 finally 關(guān)鍵字,其實(shí)這個(gè)并沒有什么可多說的,finally 的特點(diǎn)就是不管有沒有出現(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 ...
說了這么多,最后我們來結(jié)合上述內(nèi)容來處理下除0錯(cuò)誤的異常拋出。在文章開頭已經(jīng)說過,錯(cuò)誤是應(yīng)該避免的,而異常是屬于邏輯業(yè)務(wù)的。所以當(dāng)我們接到一個(gè)需要做除法的參數(shù)時(shí),可以先判斷這個(gè)數(shù)是否為0,如果是0的話,就拋出異常讓上層調(diào)用者來處理,如果不是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)的使用就是這些了,通過這兩篇文章,相信大家已經(jīng)對(duì)PHP的錯(cuò)誤和異常有了一些直觀的了解了。接下來的文章我們將一起對(duì)比下錯(cuò)誤和異常,并且說明一下PHP7對(duì)錯(cuò)誤有了哪些改進(jìn)。內(nèi)容依然精彩,值得期待哦??!
以上就是PHP中的異常及其處理機(jī)制的詳細(xì)內(nèi)容,更多關(guān)于PHP 異常的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
php curl發(fā)起get與post網(wǎng)絡(luò)請(qǐng)求案例詳解
這篇文章主要介紹了php curl發(fā)起get與post網(wǎng)絡(luò)請(qǐng)求案例詳解,本篇文章通過簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-09-09Thinkphp實(shí)現(xiàn)自動(dòng)驗(yàn)證和自動(dòng)完成
這篇文章主要介紹了Thinkphp實(shí)現(xiàn)自動(dòng)驗(yàn)證和自動(dòng)完成的相關(guān)資料,需要的朋友可以參考下2015-12-12利用PHP函數(shù)計(jì)算中英文字符串長(zhǎng)度的方法
這篇文章主要介紹了利用PHP函數(shù)計(jì)算中英文字符串長(zhǎng)度的方法,實(shí)例對(duì)比了PHP函數(shù)實(shí)現(xiàn)方法與正則表達(dá)式的實(shí)現(xiàn)方法,具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2014-11-11