上回文章中我們講到了錯誤是編譯和語法運行時會出現(xiàn)的,它們與邏輯無關(guān),是程序員在碼代碼時不應(yīng)該出現(xiàn)的,也就是說,這些錯誤應(yīng)該是盡量避免帶到線上環(huán)境的,他們不能通過try...catch捕獲到。而異常則正好相反。
什么是異常?
異常,指的是程序運行中出現(xiàn)的不符合預(yù)期的情況,通常允許它發(fā)生,并交由相應(yīng)的異常處理來進行處理。當然,你也可以選擇忽略掉異常的處理,但是就像嚴重錯誤一樣,代碼馬上會終止運行。異常屬于業(yè)務(wù)邏輯上的錯誤,基本上是我們?nèi)藶榈摹?/p>
還是先通過一個簡單的代碼看下異常的拋出和捕獲:
function test()
{
throw new Exception('This is test Error...');
}
try {
test();
} catch (Exception $e) {
print_r($e);
}
我們通過 throw 來拋出異常,然后在調(diào)用方法時將方法包裹在 try...catch 塊中來捕獲拋出的異常。這就是異常最基礎(chǔ)的結(jié)構(gòu)。
從這里我們可以看出,異?;径际峭ㄟ^我們手動進行拋出的,讓外部來進行處理。在PHP內(nèi)部多數(shù)也是在類中會進行異常的拋出,這就是面向?qū)ο蟮腻e誤處理思想了。比如說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ù)是錯誤,是無法捕獲的。而傳了的參數(shù)不對,就是異常了,在PDO類的源碼中發(fā)現(xiàn)參數(shù)不對進行了拋出。交給上層代碼也就是我們這些調(diào)用方來進行捕獲。
接下來,我們看下自定義的異常類和finally語句塊的使用。
自定義的異常類都會去繼承 Exception 類,這個類可以看做是所有異常的基類。它的結(jié)構(gòu)如下:
class Exception
{
protected $message = 'Unknown exception'; // 異常信息
private $string; // __toString cache
protected $code = 0; // 用戶自定義異常代碼
protected $file; // 發(fā)生異常的文件名
protected $line; // 發(fā)生異常的代碼行號
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)生致命錯誤
final public function getMessage(); // 返回異常信息
final public function getCode(); // 返回異常代碼
final public function getFile(); // 返回發(fā)生異常的文件名
final public function getLine(); // 返回發(fā)生異常的代碼行號
final public function getTrace(); // backtrace() 數(shù)組
final public function getPrevious(); // 之前的 exception
final public function getTraceAsString(); // 已格成化成字符串的 getTrace() 信息
// Overrideable
public function __toString(); // 可輸出的字符串
}
注意上面那行注釋的代碼,沒有傳參數(shù)是錯誤,是無法捕獲的。而傳了的參數(shù)不對,就是異常了,在PDO類的源碼中發(fā)現(xiàn)參數(shù)不對進行了拋出。交給上層代碼也就是我們這些調(diào)用方來進行捕獲。
接下來,我們看下自定義的異常類和finally語句塊的使用。
自定義的異常類都會去繼承 Exception 類,這個類可以看做是所有異常的基類。它的結(jié)構(gòu)如下:
class Exception
{
protected $message = 'Unknown exception'; // 異常信息
private $string; // __toString cache
protected $code = 0; // 用戶自定義異常代碼
protected $file; // 發(fā)生異常的文件名
protected $line; // 發(fā)生異常的代碼行號
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)生致命錯誤
final public function getMessage(); // 返回異常信息
final public function getCode(); // 返回異常代碼
final public function getFile(); // 返回發(fā)生異常的文件名
final public function getLine(); // 返回發(fā)生異常的代碼行號
final public function getTrace(); // backtrace() 數(shù)組
final public function getPrevious(); // 之前的 exception
final public function getTraceAsString(); // 已格成化成字符串的 getTrace() 信息
// Overrideable
public function __toString(); // 可輸出的字符串
}
通過上述類定義,我們可以看出,我們能重寫 構(gòu)造函數(shù) 和 __toString() 方法,也能使用一些受保護的屬性。那么我們就來定義一個自定義的異常類吧。
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框架都會有自定義異常的組件或者能力供我們使用,因為現(xiàn)代框架還是以面向?qū)ο鬄榛A(chǔ)的,所以異常會定義的比較詳細。不同組件會提供不同的異常類來進行異常的提示封裝。
接下來就是 finally 關(guān)鍵字,其實這個并沒有什么可多說的,finally 的特點就是不管有沒有出現(xiàn)異常,都會去執(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錯誤的異常拋出。在文章開頭已經(jīng)說過,錯誤是應(yīng)該避免的,而異常是屬于邏輯業(yè)務(wù)的。所以當我們接到一個需要做除法的參數(shù)時,可以先判斷這個數(shù)是否為0,如果是0的話,就拋出異常讓上層調(diào)用者來處理,如果不是0的話,就讓它正常進行除法運算就好了。
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的錯誤和異常有了一些直觀的了解了。接下來的文章我們將一起對比下錯誤和異常,并且說明一下PHP7對錯誤有了哪些改進。內(nèi)容依然精彩,值得期待哦??!
以上就是PHP中的異常及其處理機制的詳細內(nèi)容,更多關(guān)于PHP 異常的資料請關(guān)注腳本之家其它相關(guān)文章!
您可能感興趣的文章:- 再談PHP錯誤與異常處理
- PHP中的異常處理機制深入講解
- php中try catch捕獲異常實例詳解
- Thinkphp5框架異常處理操作實例分析
- 讓whoops幫我們告別ThinkPHP6的異常頁面
- Laravel 解決composer相關(guān)操作提示php相關(guān)異常的問題
- Thinkphp 在api開發(fā)中異常返回依然是html的解決方式
- PHP使用觀察者模式處理異常信息的方法詳解
- php異常處理捕獲錯誤整理
- PHP批斗大會之缺失的異常詳解