異常處理(又稱為錯誤處理)功能提供了處理程序運行時出現的錯誤或異常情況的方法。
異常處理通常是防止未知錯誤產生所采取的處理措施。異常處理的好處是你不用再絞盡腦汁去考慮各種錯誤,這為處理某一類錯誤提供了一個很有效的方法,使編程效率大大提高。當異常被觸發時,通常會發生:
當前代碼狀態被保存
代碼執行被切換到預定義的異常處理器函數
根據情況,處理器也許會從保存的代碼狀態重新開始執行代碼,終止腳本執行,或從代碼中另外的位置繼續執行腳本
PHP 5 提供了一種新的面向對象的錯誤處理方法。可以使用檢測(try)、拋出(throw)和捕獲(catch)異常。即使用try檢測有沒有拋出(throw)異常,若有異常拋出(throw),使用catch捕獲異常。
一個 try 至少要有一個與之對應的 catch。定義多個 catch 可以捕獲不同的對象。PHP 會按這些 catch 被定義的順序執行,直到完成最後一個為止。而在這些 catch 內,又可以拋出新的異常。
1. 異常的使用
當一個異常被拋出時,其後的代碼將不會繼續執行,PHP 會嘗試查找匹配的 "catch" 代碼塊。如果一個異常沒有被捕獲,而且又沒用使用set_exception_handler() 作相應的處理的話,那麼 PHP 將會產生一個嚴重的錯誤,並且輸出未能捕獲異常(Uncaught Exception ... )的提示信息。
拋出異常,但不去捕獲它:
<?php
ini_set('display_errors', 'On');
error_reporting(E_ALL & ~ E_WARNING);
$error = 'Always throw this error';
throw new Exception($error);
// 繼續執行
echo 'Hello World';
?>
上面的代碼會獲得類似這樣的一個致命錯誤:
Fatal error: Uncaught exception 'Exception' with message 'Always throw this error' in E:\sngrep\index.php on line 5
Exception: Always throw this error in E:\sngrep\index.php on line 5
Call Stack:
0.0005 330680 1. {main}() E:\sngrep\index.php:0
2. Try, throw 和 catch
要避免上面這個致命錯誤,可以使用try catch捕獲掉。
處理處理程序應當包括:
Try - 使用異常的函數應該位於 "try" 代碼塊內。如果沒有觸發異常,則代碼將照常繼續執行。但是如果異常被觸發,會拋出一個異常。
Throw - 這裡規定如何觸發異常。每一個 "throw" 必須對應至少一個 "catch"
Catch - "catch" 代碼塊會捕獲異常,並創建一個包含異常信息的對象
拋出異常並捕獲掉,可以繼續執行後面的代碼:
<?php
try {
$error = 'Always throw this error';
throw new Exception($error);
// 從這裡開始,tra 代碼塊內的代碼將不會被執行
echo 'Never executed';
} catch (Exception $e) {
echo 'Caught exception: ', $e->getMessage(),'<br>';
}
// 繼續執行
echo 'Hello World';
?>
在 "try" 代碼塊檢測有有沒有拋出“throw”異常,這裡拋出了異常。
"catch" 代碼塊接收到該異常,並創建一個包含異常信息的對象 ($e)。
通過從這個 exception 對象調用 $e->getMessage(),輸出來自該異常的錯誤消息
為了遵循“每個 throw 必須對應一個 catch”的原則,可以設置一個頂層的異常處理器來處理漏掉的錯誤。
3. 擴展 PHP 內置的異常處理類
用戶可以用自定義的異常處理類來擴展 PHP 內置的異常處理類。以下的代碼說明了在內置的異常處理類中,哪些屬性和方法在子類中是可訪問和可繼承的。(注:以下這段代碼只為說明內置異常處理類的結構,它並不是一段有實際意義的可用代碼。)
<?php
class Exception
{
protected $message = 'Unknown exception'; // 異常信息
protected $code = 0; // 用戶自定義異常代碼
protected $file; // 發生異常的文件名
protected $line; // 發生異常的代碼行號
function __construct($message = null, $code = 0);
final function getMessage(); // 返回異常信息
final function getCode(); // 返回異常代碼
final function getFile(); // 返回發生異常的文件名
final function getLine(); // 返回發生異常的代碼行號
final function getTrace(); // backtrace() 數組 www.2cto.com
final function getTraceAsString(); // 已格成化成字符串的 getTrace() 信息
/* 可重載的方法 */
function __toString(); // 可輸出的字符串
}
如果使用自定義的類來擴展內置異常處理類,並且要重新定義構造函數的話,建議同時調用 parent::__construct() 來檢查所有的變量是否已被賦值。當對象要輸出字符串的時候,可以重載__toString() 並自定義輸出的樣式。
構建自定義異常處理類:
<?php
/**
*
* 自定義一個異常處理類
*/
class MyException extends Exception
{
// 重定義構造器使 message 變為必須被指定的屬性
public function __construct($message, $code = 0) {
// 自定義的代碼
// 確保所有變量都被正確賦值
parent::__construct($message, $code);
}
// 自定義字符串輸出的樣式 */
public function __toString() {
return __CLASS__ . ": [{$this->code}]: {$this->message}\n";
}
public function customFunction() {
echo "A Custom function for this type of exception\n";
}
}
// 例子 1:拋出自定義異常,但沒有默認的異常
echo ' 例子 1', '<br>';
try {
// 拋出自定義異常
throw new MyException('1 is an invalid parameter', 5);
} catch (MyException $e) { // 捕獲異常
echo "Caught my exception\n", $e;
$e->customFunction();
} catch (Exception $e) { // 被忽略
echo "Caught Default Exception\n", $e;
}
// 執行後續代碼
// 例子 2: 拋出默認的異常 但沒有自定義異常
echo '<br>', ' 例子 2:', '<br>';
try {
// 拋出默認的異常
throw new Exception('2 isnt allowed as a parameter', 6);
} catch (MyException $e) { // 不能匹配異常的種類,被忽略
echo "Caught my exception\n", $e;
$e->customFunction();
} catch (Exception $e) {// 捕獲異常
echo "Caught Default Exception\n", $e;
}
// 執行後續代碼
// 例子 3: 拋出自定義異常 ,使用默認異常類對象來捕獲
echo '<br>', ' 例子 3:', '<br>';
try {
// 拋出自定義異常
throw new MyException('3 isnt allowed as a parameter', 6);
} catch (Exception $e) { // 捕獲異常
echo "Default Exception caught\n", $e;
}
// 執行後續代碼
// 例子 4
echo '<br>', ' 例子 4:', '<br>';
try {
echo 'No Exception ';
} catch (Exception $e) { // 沒有異常,被忽略
echo "Default Exception caught\n", $e;
}
// 執行後續代碼
MyException 類是作為舊的 exception 類的一個擴展來創建的。這樣它就繼承了舊類的所有屬性和方法,我們可以使用 exception 類的方法,比如 getLine() 、 getFile() 以及 getMessage()。
4. 嵌套異常處理
如果在內層 "try" 代碼塊中異常沒有被捕獲,則它將在外層級上查找 catch 代碼塊去捕獲。
try {
try {
throw new MyException('foo!');
} catch (MyException $e) {
/* 重新拋出 rethrow it */
$e->customFunction();
throw $e;
}
} catch (Exception $e) {
var_dump($e->getMessage());
}
5. 設置頂層異常處理器 (Top Level Exception Handler)
set_exception_handler() 函數可設置處理所有未捕獲異常的用戶定義函數。
<?php
function myException($exception)
{
echo "<b>Exception:</b> " , $exception->getMessage();
}
set_exception_handler('myException');
throw new Exception('Uncaught Exception occurred');
輸出結果:
Exception: Uncaught Exception occurred
6. 異常的規則
需要進行異常處理的代碼應該放入 try 代碼塊內,以便捕獲潛在的異常。
每個 try 或 throw 代碼塊必須至少擁有一個對應的 catch 代碼塊。
使用多個 catch 代碼塊可以捕獲不同種類的異常。
可以在 try 代碼塊內的 catch 代碼塊中再次拋出(re-thrown)異常。
簡而言之:如果拋出了異常,就必須捕獲它,否則程序終止執行。
摘自 程序人生,guisu專欄