在編寫php程序時,錯誤處理是一個重要的部分。如果程序中缺少錯誤檢測代碼,那么看上去很不專業,也為安全風險敞開了大門
錯誤報告級別(了解即可)
這些錯誤報告級別是錯誤處理程序旨在處理的錯誤的不同的類型:
值 | 常量 | 描述 |
---|---|---|
2 | E_WARNING | 非致命的 run-time 錯誤。不暫停腳本執行。 |
8 | E_NOTICE | Run-time 通知。 腳本發現可能有錯誤發生,但也可能在腳本正常運行時發生。 |
256 | E_USER_ERROR | 致命的用戶生成的錯誤。這類似於程序員使用 PHP 函數 trigger_error() 設置的 E_ERROR。 |
512 | E_USER_WARNING | 非致命的用戶生成的警告。這類似於程序員使用 PHP 函數 trigger_error() 設置的 E_WARNING。 |
1024 | E_USER_NOTICE | 用戶生成的通知。這類似於程序員使用 PHP 函數 trigger_error() 設置的 E_NOTICE。 |
4096 | E_RECOVERABLE_ERROR | 可捕獲的致命錯誤。類似 E_ERROR,但可被用戶定義的處理程序捕獲。(參見 set_error_handler()) |
8191 | E_ALL | 所有錯誤和警告,除級別 E_STRICT 以外。 (在 PHP 6.0,E_STRICT 是 E_ALL 的一部分)
|
需求:比如要接收一個年齡,如果數字大於120,就認為是一個錯誤傳統方法:
if($age>120){
echo '年齡錯誤';exit();
}
1 <?php 2 date_default_timezone_set('PRC'); 3 function myerror($error_level,$error_message){ 4 $info= "錯誤號:$error_level\n"; 5 $info.= "錯誤信息:$error_message\n"; 6 $info.= '發生時間:'.date('Y-m-d H:i:s'); 7 $filename='aa.txt'; 8 if(!$fp=fopen($filename,'a')){ 9 '創建文件'.$filename.'失敗'; 10 } 11 if(is_writeable($filename)){ 12 if(!fwrite($fp,$info)){ 13 echo '寫入文件失敗'; 14 } else { 15 echo '已成功記錄錯誤信息'; 16 } 17 fclose($fp); 18 } else { 19 echo '文件'.$filename.'不可寫'; 20 } 21 exit(); 22 } 23 set_error_handler('myerror',E_WARNING); 24 $fp=fopen('aaa.txt','r'); 25 ?>
- Try - 使用異常的函數應該位於 "try" 代碼塊內。如果沒有觸發異常,則代碼將照常繼續執行。但是如果異常被觸發,會拋出一個異常。
- Throw - 這里規定如何觸發異常。每一個 "throw" 必須對應至少一個 "catch"
- Catch - "catch" 代碼塊會捕獲異常,並創建一個包含異常信息的對象
讓我們觸發一個異常:
1 <?php 2 //創建可拋出一個異常的函數 3 function checkNum($number){ 4 if($number>1){ 5 throw new Exception("Value must be 1 or below"); 6 } 7 return true; 8 } 9 10 //在 "try" 代碼塊中觸發異常 11 try{ 12 checkNum(2); 13 //如果異常被拋出,那么下面一行代碼將不會被輸出 14 echo 'If you see this, the number is 1 or below'; 15 }catch(Exception $e){ 16 //捕獲異常 17 echo 'Message: ' .$e->getMessage(); 18 } 19 ?>
上面代碼將獲得類似這樣一個錯誤:
Message: Value must be 1 or below
例子解釋:
上面的代碼拋出了一個異常,並捕獲了它:
- 創建 checkNum() 函數。它檢測數字是否大於 1。如果是,則拋出一個異常。
- 在 "try" 代碼塊中調用 checkNum() 函數。
- checkNum() 函數中的異常被拋出
- "catch" 代碼塊接收到該異常,並創建一個包含異常信息的對象 ($e)。
- 通過從這個 exception 對象調用 $e->getMessage(),輸出來自該異常的錯誤消息
不過,為了遵循“每個 throw 必須對應一個 catch”的原則,可以設置一個頂層的異常處理器來處理漏掉的錯誤。
set_exception_handler()函數可設置處理所有未捕獲異常的用戶定義函數
//設置一個頂級異常處理器
function myexception($e){
echo 'this is top exception';
} //修改默認的異常處理器
set_exception_handler("myexception");
try{
$i=5;
if($i<10){
throw new exception('$i must greater than 10');
}
}catch(Exception $e){
//處理異常
echo $e->getMessage().'<br/>';
//不處理異常,繼續拋出
throw new exception('errorinfo'); //也可以用throw $e 保留原錯誤信息;
}
創建一個自定義的異常類
class customException extends Exception{
public function errorMessage(){
//error message $errorMsg = 'Error on line '.$this->getLine().' in '.$this->getFile().': <b>'.$this->getMessage().'</b> is not a valid E-Mail address'; return $errorMsg;
}
}
//使用
try{
throw new customException('error message');
}catch(customException $e){
echo $e->errorMsg();
}
可以使用多個catch來返回不同情況下的錯誤信息
try{
$i=5;
if($i>0){
throw new customException('error message');//使用自定義異常類處理
} if($i<-10){
throw new exception('error2');//使用系統默認異常處理
}
}catch(customException $e){
echo $e->getMessage();
}catch(Exception $e1){
echo $e1->getMessage();
}
異常的規則
- 需要進行異常處理的代碼應該放入 try 代碼塊內,以便捕獲潛在的異常。
- 每個try或throw代碼塊必須至少擁有一個對應的 catch 代碼塊。
- 使用多個 catch 代碼塊可以捕獲不同種類的異常。
- 可以在try代碼內的catch 代碼塊中再次拋出(re-thrown)異常。
簡而言之:如果拋出了異常,就必須捕獲它。