php異常處理

PHP異常處理詳解

         異常處理(又稱爲錯誤處理)功能提供了處理程序運行時出現的錯誤或異常狀況的方法。php

  異常處理一般是防止未知錯誤產生所採起的處理措施。異常處理的好處是你不用再絞盡腦汁去考慮各類錯誤,這爲處理某一類錯誤提供了一個頗有效的方法,使編程效率大大提升。當異常被觸發時,一般會發生:
          當前代碼狀態被保存
         代碼執行被切換到預約義的異常處理器函數
          根據狀況,處理器也許會從保存的代碼狀態從新開始執行代碼,終止腳本執行,或從代碼中另外的位置繼續執行腳本
java

          PHP 5 提供了一種新的面向對象的錯誤處理方法。可使用檢測(try)、拋出(throw)和捕獲(catch)異常。即便用try檢測有沒有拋出(throw)異常,如有異常拋出(throw),使用catch捕獲異常。web

         一個 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';  
?>

上面的代碼會得到相似這樣的一個致命錯誤:
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

2. Try, throw 和 catch

要避免上面這個致命錯誤,可使用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」的原則,能夠設置一個頂層的異常處理器來處理漏掉的錯誤。

3. 擴展 PHP 內置的異常處理類

    用戶能夠用自定義的異常處理類來擴展 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()。

4. 嵌套異常處理

    若是在內層 "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

6. 異常的規則

  • 須要進行異常處理的代碼應該放入 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),配合使用。

相關文章
相關標籤/搜索