程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> PHP編程 >> 關於PHP編程 >> PHP異常處理詳解

PHP異常處理詳解

編輯:關於PHP編程

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

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

          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專欄

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved