異常處理(又稱爲錯誤處理)功能提供了處理程序運行時出現的錯誤或異常狀況的方法。php
異常處理一般是防止未知錯誤產生所採起的處理措施。異常處理的好處是你不用再絞盡腦汁去考慮各類錯誤,這爲處理某一類錯誤提供了一個頗有效的方法,使編程效率大大提升。當異常被觸發時,一般會發生:
當前代碼狀態被保存
代碼執行被切換到預約義的異常處理器函數
根據狀況,處理器也許會從保存的代碼狀態從新開始執行代碼,終止腳本執行,或從代碼中另外的位置繼續執行腳本java
PHP 5 提供了一種新的面向對象的錯誤處理方法。可使用檢測(try)、拋出(throw)和捕獲(catch)異常。即便用try檢測有沒有拋出(throw)異常,如有異常拋出(throw),使用catch捕獲異常。web
一個 try 至少要有一個與之對應的 catch。定義多個 catch 能夠捕獲不一樣的對象。PHP 會按這些 catch 被定義的順序執行,直到完成最後一個爲止。而在這些 catch 內,又能夠拋出新的異常。編程
當一個異常被拋出時,其後的代碼將不會繼續執行,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'; ?>
上面的代碼會得到相似這樣的一個致命錯誤:
ui
[plain] view plaincopyprint? 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
要避免上面這個致命錯誤,可使用try catch捕獲掉。
this
處理處理程序應當包括:
Try - 使用異常的函數應該位於 "try" 代碼塊內。若是沒有觸發異常,則代碼將照常繼續執行。可是若是異常被觸發,會拋出一個異常。
Throw - 這裏規定如何觸發異常。每個 "throw" 必須對應至少一個 "catch"
Catch - "catch" 代碼塊會捕獲異常,並建立一個包含異常信息的對象 spa
拋出異常並捕獲掉,能夠繼續執行後面的代碼:.net
[php] view plaincopyprint? <?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」的原則,能夠設置一個頂層的異常處理器來處理漏掉的錯誤。
用戶能夠用自定義的異常處理類來擴展 PHP 內置的異常處理類。如下的代碼說明了在內置的異常處理類中,哪些屬性和方法在子類中是可訪問和可繼承的。(注:如下這段代碼只爲說明內置異常處理類的結構,它並非一段有實際意義的可用代碼。)
[php] view plaincopyprint? <?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() 數組 final function getTraceAsString(); // 已格成化成字符串的 getTrace() 信息 /* 可重載的方法 */ function __toString(); // 可輸出的字符串 }
若是使用自定義的類來擴展內置異常處理類,而且要從新定義構造函數的話,建議同時調用parent::__construct() 來檢查全部的變量是否已被賦值。當對象要輸出字符串的時候,能夠重載__toString() 並自定義輸出的樣式。
構建自定義異常處理類:
[php] view plaincopyprint? <?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()。
若是在內層 "try" 代碼塊中異常沒有被捕獲,則它將在外層級上查找 catch 代碼塊去捕獲。
[php] view plaincopyprint? try { try { throw new MyException('foo!'); } catch (MyException $e) { /* 從新拋出 rethrow it */ $e->customFunction(); throw $e; } } catch (Exception $e) { var_dump($e->getMessage()); }
set_exception_handler() 函數可設置處理全部未捕獲異常的用戶定義函數。
[php] view plaincopyprint? <?php function myException($exception) { echo "<b>Exception:</b> " , $exception->getMessage(); } set_exception_handler('myException'); throw new Exception('Uncaught Exception occurred');
[php] view plaincopyprint? Exception: Uncaught Exception occurred
須要進行異常處理的代碼應該放入 try 代碼塊內,以便捕獲潛在的異常。
每一個 try 或 throw 代碼塊必須至少擁有一個對應的 catch 代碼塊。
使用多個 catch 代碼塊能夠捕獲不一樣種類的異常。
能夠在 try 代碼塊內的 catch 代碼塊中再次拋出(re-thrown)異常。
簡而言之:若是拋出了異常,就必須捕獲它,不然程序終止執行。
在咱們實際開發中,錯誤及異常捕捉僅僅靠try{}catch()是遠遠不夠的。
set_error_handler : 通常用於捕捉 E_NOTICE 、E_USER_ERROR、E_USER_WARNING、E_USER_NOTICE
不能捕捉:E_ERROR, E_PARSE, E_CORE_ERROR, E_CORE_WARNING, E_COMPILE_ERROR and E_COMPILE_WARNING。通常與trigger_error("...", E_USER_ERROR),配合使用。