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è)簡單的代碼看下異常的拋出和捕獲:
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ù)不對,就是異常了,在PDO類的源碼中發(fā)現(xiàn)參數(shù)不對進(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ù)不對,就是異常了,在PDO類的源碼中發(fā)現(xiàn)參數(shù)不對進(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)對PHP的錯(cuò)誤和異常有了一些直觀的了解了。接下來的文章我們將一起對比下錯(cuò)誤和異常,并且說明一下PHP7對錯(cuò)誤有了哪些改進(jìn)。內(nèi)容依然精彩,值得期待哦!!
以上就是PHP中的異常及其處理機(jī)制的詳細(xì)內(nèi)容,更多關(guān)于PHP 異常的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. 基于PHP做個(gè)圖片防盜鏈2. ASP.NET MVC把數(shù)據(jù)庫中枚舉項(xiàng)的數(shù)字轉(zhuǎn)換成文字3. asp.net core 認(rèn)證和授權(quán)實(shí)例詳解4. XML在語音合成中的應(yīng)用5. .NET中實(shí)現(xiàn)對象數(shù)據(jù)映射示例詳解6. 基于javaweb+jsp實(shí)現(xiàn)企業(yè)車輛管理系統(tǒng)7. ASP.NET MVC使用Boostrap實(shí)現(xiàn)產(chǎn)品展示、查詢、排序、分頁8. 如何使用ASP.NET Core 配置文件9. jscript與vbscript 操作XML元素屬性的代碼10. php使用正則驗(yàn)證密碼字段的復(fù)雜強(qiáng)度原理詳細(xì)講解 原創(chuàng)
