異常處理(又稱為錯誤處理)功能提供了處理常式運行時出現的錯誤或異常情況的方法。
異常處理通常是防止未知錯誤產生所採取的處理措施。異常處理的好處是你不用再絞盡腦汁去考慮各種錯誤,這為處理某一類錯誤提供了一個很有效方法,使編程效率大大提高。當異常被觸發時,通常會發生:
當前代碼狀態被儲存
代碼執行被切換到預定義的異常處理器函數
根據情況,處理器也許會從儲存的代碼狀態重新開始執行代碼,終止指令碼執行,或從代碼中另外的位置繼續執行指令碼
PHP 5 提供了一種新的物件導向的錯誤處理方法。可以使用檢測(try)、拋出(throw)和捕獲(catch)異常。即使用try檢測有沒有拋出(throw)異常,若有異常拋出(throw),使用catch捕獲異常。
一個 try 至少要有一個與之對應的 catch。定義多個 catch 可以捕獲不同的對象。PHP 會按這些 catch 被定義的順序執行,直到完成最後一個為止。而在這些 catch 內,又可以拋出新的異常。
1. 異常的使用
當一個異常被拋出時,其後的代碼將不會繼續執行,PHP 會嘗試尋找匹配的 "catch" 代碼塊。如果一個異常沒有被捕獲,而且又沒用使用set_exception_handler() 作相應的處理的話,那麼 PHP 將會產生一個嚴重的錯誤,並且輸出未能捕獲異常(Uncaught Exception ... )的提示資訊。
拋出異常,但不去捕獲它:
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" 代碼塊會捕獲異常,並建立一個包含異常資訊的對象
拋出異常並捕獲掉,可以繼續執行後面的代碼:
try {
$error = 'Always throw this error';
throw new Exception($error);
// 從這裡開始,tra 代碼塊內的代碼將不會被執行
echo 'Never executed';
} catch (Exception $e) {
echo 'Caught exception: ', $e->getMessage(),'
';
}
// 繼續執行
echo 'Hello World';
?>
在 "try" 代碼塊檢測有有沒有拋出“throw”異常,這裡拋出了異常。
"catch" 代碼塊接收到該異常,並建立一個包含異常資訊的對象 ($e)。
通過從這個 exception 對象調用 $e->getMessage(),輸出來自該異常的錯誤訊息
為了遵循“每個 throw 必須對應一個 catch”的原則,可以設定一個頂層的異常處理器來處理漏掉的錯誤。
3. 擴充 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() 並自訂輸出的樣式。
構建自訂異常處理類:
/**
*
* 自訂一個異常處理類
*/
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', '
';
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 '
', ' 例子 2:', '
';
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 '
', ' 例子 3:', '
';
try {
// 拋出自訂異常
throw new MyException('3 isnt allowed as a parameter', 6);
} catch (Exception $e) { // 捕獲異常
echo "Default Exception caught\n", $e;
}
// 執行後續代碼
// 例子 4
echo '
', ' 例子 4:', '
';
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() 函數可設定處理所有未捕獲異常的使用者定義函數。
function myException($exception)
{
echo "Exception: " , $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專欄
http://www.bkjia.com/PHPjc/478292.htmlwww.bkjia.comtruehttp://www.bkjia.com/PHPjc/478292.htmlTechArticle異常處理(又稱為錯誤處理)功能提供了處理常式運行時出現的錯誤或異常情況的方法。 異常處理通常是防止未知錯誤產生所採取的處理措...