亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > PHP > 正文

PHP各種異常和錯誤的攔截方法及發生致命錯誤時進行報警

2024-05-04 23:42:20
字體:
來源:轉載
供稿:網友

在日常開發中,大多數人的做法是在開發環境時開啟調試模式,在產品環境關閉調試模式。在開發的時候可以查看各種錯誤、異常,但是在線上就把錯誤顯示的關閉。

上面的情形看似很科學,有人解釋為這樣很安全,別人看不到錯誤,以免泄露重要信息...

但是你有沒有遇到這種情況,線下好好的,一上線卻運行不起來也找不到原因...

一個腳本,跑了好長一段時間,一直沒有問題,有一天突然中斷了,然后了也沒有任何記錄都不造啥原因...

線上一個付款,別人明明付了款,但是我們卻沒有記錄到,自己親自去實驗,卻是好的...

種種以上,都是因為大家關閉了錯誤信息,并且未將錯誤、異常記錄到日志,導致那些隨機發生的錯誤很難追蹤。這樣矛盾就來了,即不要顯示錯誤,又要追蹤錯誤,這如何實現了?

以上問題都可以通過PHP的錯誤、異常機制及其內建函數'set_exception_handler','set_error_handler','register_shutdown_function' 來實現

'set_exception_handler' 函數 用于攔截各種未捕獲的異常,然后將這些交給用戶自定義的方式進行處理

'set_error_handler' 函數可以攔截各種錯誤,然后交給用戶自定義的方式進行處理

'register_shutdown_function' 函數是在PHP腳本結束時調用的函數,配合'error_get_last'可以獲取最后的致命性錯誤

這個思路大體就是把錯誤、異常、致命性錯誤攔截下來,交給我們自定義的方法進行處理,我們辨別這些錯誤、異常是否致命,如果是則記錄的數據庫或者文件系統,然后使用腳本不停的掃描這些日志,發現嚴重錯誤立即發送郵件或發送短信進行報警

首先我們定義錯誤攔截類,該類用于將錯誤、異常攔截下來,用我們自己定義的處理方式進行處理,該類放在文件名為'errorHandler.class.php'中,代碼如下

/** * 文件名稱:baseErrorHandler.class.php * 摘 要:錯誤攔截器父類 */require 'errorHandlerException.class.php';//異常類class errorHandler{ public $argvs = array(); public  $memoryReserveSize = 262144;//備用內存大小 private $_memoryReserve;//備用內存 /**  * 方  法:注冊自定義錯誤、異常攔截器  * 參  數:void  * 返  回:void  */ public function register() {  ini_set('display_errors', 0);  set_exception_handler(array($this, 'handleException'));//截獲未捕獲的異常  set_error_handler(array($this, 'handleError'));//截獲各種錯誤 此處切不可掉換位置  //留下備用內存 供后面攔截致命錯誤使用  $this->memoryReserveSize > 0 && $this->_memoryReserve = str_repeat('x', $this->memoryReserveSize);  register_shutdown_function(array($this, 'handleFatalError'));//截獲致命性錯誤 } /**  * 方  法:取消自定義錯誤、異常攔截器  * 參  數:void  * 返  回:void  */ public function unregister() {  restore_error_handler();  restore_exception_handler(); } /**  * 方  法:處理截獲的未捕獲的異常  * 參  數:Exception $exception  * 返  回:void  */ public function handleException($exception) {  $this->unregister();  try  {   $this->logException($exception);   exit(1);  }  catch(Exception $e)  {   exit(1);  } } /**  * 方  法:處理截獲的錯誤  * 參  數:int  $code 錯誤代碼  * 參  數:string $message 錯誤信息  * 參  數:string $file 錯誤文件  * 參  數:int  $line 錯誤的行數  * 返  回:boolean  */ public function handleError($code, $message, $file, $line) {  //該處思想是將錯誤變成異常拋出 統一交給異常處理函數進行處理  if((error_reporting() & $code) && !in_array($code, array(E_NOTICE, E_WARNING, E_USER_NOTICE, E_USER_WARNING, E_DEPRECATED)))  {//此處只記錄嚴重的錯誤 對于各種WARNING NOTICE不作處理   $exception = new errorHandlerException($message, $code, $code, $file, $line);   $trace = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS);   array_shift($trace);//trace的第一個元素為當前對象 移除   foreach($trace as $frame)    {    if($frame['function'] == '__toString')     {//如果錯誤出現在 __toString 方法中 不拋出任何異常     $this->handleException($exception);     exit(1);    }   }   throw $exception;  }  return false; } /**  * 方  法:截獲致命性錯誤  * 參  數:void  * 返  回:void  */ public function handleFatalError() {  unset($this->_memoryReserve);//釋放內存供下面處理程序使用  $error = error_get_last();//最后一條錯誤信息  if(errorHandlerException::isFatalError($error))  {//如果是致命錯誤進行處理   $exception = new errorHandlerException($error['message'], $error['type'], $error['type'], $error['file'], $error['line']);   $this->logException($exception);   exit(1);  } } /**  * 方  法:獲取服務器IP  * 參  數:void  * 返  回:string  */ final public function getServerIp() {  $serverIp = '';  if(isset($_SERVER['SERVER_ADDR']))  {   $serverIp = $_SERVER['SERVER_ADDR'];  }  elseif(isset($_SERVER['LOCAL_ADDR']))  {   $serverIp = $_SERVER['LOCAL_ADDR'];  }  elseif(isset($_SERVER['HOSTNAME']))  {   $serverIp = gethostbyname($_SERVER['HOSTNAME']);  }  else  {   $serverIp = getenv('SERVER_ADDR');  }    return $serverIp;  } /**  * 方  法:獲取當前URI信息  * 參  數:void  * 返  回:string $url  */ public function getCurrentUri() {  $uri = '';  if($_SERVER ["REMOTE_ADDR"])  {//瀏覽器瀏覽模式   $uri = 'http://' . $_SERVER['SERVER_NAME'] . $_SERVER['REQUEST_URI'];  }  else  {//命令行模式   $params = $this->argvs;   $uri = $params[0];   array_shift($params);   for($i = 0, $len = count($params); $i < $len; $i++)   {    $uri .= ' ' . $params[$i];   }  }  return $uri; } /**  * 方  法:記錄異常信息  * 參  數:errorHandlerException $e 錯誤異常  * 返  回:boolean 是否保存成功  */ final public function logException($e) {  $error = array(      'add_time'  =>  time(),      'title'  =>  errorHandlerException::getName($e->getCode()),//這里獲取用戶友好型名稱      'message'  =>  array(),      'server_ip' =>  $this->getServerIp(),      'code'   =>  errorHandlerException::getLocalCode($e->getCode()),//這里為各種錯誤定義一個編號以便查找      'file'   => $e->getFile(),      'line'   =>  $e->getLine(),      'url'  => $this->getCurrentUri(),     );  do  {   //$e->getFile() . ':' . $e->getLine() . ' ' . $e->getMessage() . '(' . $e->getCode() . ')'   $message = (string)$e;   $error['message'][] = $message;  } while($e = $e->getPrevious());  $error['message'] = implode("/r/n", $error['message']);  $this->logError($error); } /**  * 方  法:記錄異常信息  * 參  數:array $error = array(  *         'time' => int,   *         'title' => 'string',   *         'message' => 'string',   *         'code' => int,  *         'server_ip' => 'string'  *          'file'  => 'string',  *         'line' => int,  *         'url' => 'string',  *        );  * 返  回:boolean 是否保存成功  */ public function logError($error) {  /*這里去實現如何將錯誤信息記錄到日志*/ }}

上述代碼中,有個'errorHandlerException'類,該類放在文件'errorHandlerException.class.php'中,該類用于將錯誤轉換為異常,以便記錄錯誤發生的文件、行號、錯誤代碼、錯誤信息等信息,同時其方法'isFatalError'用于辨別該錯誤是否是致命性錯誤。這里我們為了方便管理,將錯誤進行編號并命名。該類的代碼如下

/** * 文件名稱:errorHandlerException.class.php * 摘 要:自定義錯誤異常類 該類繼承至PHP內置的錯誤異常類 */class errorHandlerException extends ErrorException{ public static $localCode = array(          E_COMPILE_ERROR => 4001,          E_COMPILE_WARNING => 4002,          E_CORE_ERROR => 4003,          E_CORE_WARNING => 4004,          E_DEPRECATED => 4005,          E_ERROR => 4006,          E_NOTICE => 4007,          E_PARSE => 4008,          E_RECOVERABLE_ERROR => 4009,          E_STRICT => 4010,          E_USER_DEPRECATED => 4011,          E_USER_ERROR => 4012,          E_USER_NOTICE => 4013,          E_USER_WARNING => 4014,          E_WARNING => 4015,          4016 => 4016,         ); public static $localName = array(          E_COMPILE_ERROR => 'PHP Compile Error',          E_COMPILE_WARNING => 'PHP Compile Warning',          E_CORE_ERROR => 'PHP Core Error',          E_CORE_WARNING => 'PHP Core Warning',          E_DEPRECATED => 'PHP Deprecated Warning',          E_ERROR => 'PHP Fatal Error',          E_NOTICE => 'PHP Notice',          E_PARSE => 'PHP Parse Error',          E_RECOVERABLE_ERROR => 'PHP Recoverable Error',          E_STRICT => 'PHP Strict Warning',          E_USER_DEPRECATED => 'PHP User Deprecated Warning',          E_USER_ERROR => 'PHP User Error',          E_USER_NOTICE => 'PHP User Notice',          E_USER_WARNING => 'PHP User Warning',          E_WARNING => 'PHP Warning',          4016 => 'Customer`s Error',         ); /**  * 方  法:構造函數  * 摘  要:相關知識請查看 http://php.net/manual/en/errorexception.construct.php  *   * 參  數:string  $message  異常信息(可選)  *    int   $code   異常代碼(可選)  *    int   $severity  *    string  $filename  異常文件(可選)  *    int   $line   異常的行數(可選)  *   Exception $previous 上一個異常(可選)  *  * 返  回:void  */ public function __construct($message = '', $code = 0, $severity = 1, $filename = __FILE__, $line = __LINE__, Exception $previous = null) {  parent::__construct($message, $code, $severity, $filename, $line, $previous); } /**  * 方  法:是否是致命性錯誤  * 參  數:array $error  * 返  回:boolean  */ public static function isFatalError($error) {  $fatalErrors = array(        E_ERROR,         E_PARSE,         E_CORE_ERROR,        E_CORE_WARNING,         E_COMPILE_ERROR,         E_COMPILE_WARNING       );  return isset($error['type']) && in_array($error['type'], $fatalErrors); } /**  * 方  法:根據原始的錯誤代碼得到本地的錯誤代碼  * 參  數:int $code  * 返  回:int $localCode  */ public static function getLocalCode($code) {  return isset(self::$localCode[$code]) ? self::$localCode[$code] : self::$localCode[4016]; } /**  * 方  法:根據原始的錯誤代碼獲取用戶友好型名稱  * 參  數:int   * 返  回:string $name  */ public static function getName($code) {  return isset(self::$localName[$code]) ? self::$localName[$code] : self::$localName[4016]; }

在錯誤攔截類中,需要用戶自己定義實現錯誤記錄的方法('logException'),這個地方需要注意,有些錯誤可能在一段時間內不斷發生,因此只需記錄一次即可,你可以使用錯誤代碼、文件、行號、錯誤詳情 生成一個MD5值用于記錄該錯誤是否已經被記錄,如果在規定時間內(一個小時)已經被記錄過則不需要再進行記錄

然后我們定義一個文件,用于實例化以上類,捕獲各種錯誤、異常,該文件命名為'registerErrorHandler.php', 內如如下

/** 使用方法介紹:* 在入口處引入該文件即可,然后可以在該文件中定義調試模式常量'DEBUG_ERROR'** <?php* * require 'registerErrorHandler.php';* * ?>*//*** 調試錯誤模式:* 0    =>   非調試模式,不顯示異常、錯誤信息但記錄異常、錯誤信息* 1    =>   調試模式,顯示異常、錯誤信息但不記錄異常、錯誤信息*/define('DEBUG_ERROR', 0);require 'errorHandler.class.php';class registerErrorHandler{ /**  * 方  法:注冊異常、錯誤攔截  * 參  數:void  * 返  回:void  */ public static function register() {  global $argv;  if(DEBUG_ERROR)  {//如果開啟調試模式   ini_set('display_errors', 1);   return;  }  //如果不開啟調試模式  ini_set('error_reporting', -1);  ini_set('display_errors', 0);  $handler = new errorHandler();  $handler->argvs = $argv;//此處主要兼容命令行模式下獲取參數  $handler->register(); } }registerErrorHandler::register(); 

剩下的就是需要你在你的入口文件引入該文件,定義調試模式,然后實現你自己記錄錯誤的方法即可

需要注意的是,有些錯誤在你進行注冊之前已經發生并且導致腳本中斷是無法記錄下來的,因為此時'registerErrorHandler::register()' 尚未執行已經中斷了

還有就是'set_error_handler'這個函數不能捕獲下面類型的錯誤 E_ERROR、 E_PARSE、 E_CORE_ERROR、 E_CORE_WARNING、E_COMPILE_ERROR、 E_COMPILE_WARNING, 這個可以在官方文檔中看到,但是本處無妨,因為以上錯誤是解析、編譯錯誤,這些都沒有通過,你是不可能發布上線的


注:相關教程知識閱讀請移步到PHP教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产一区二区三区精品久久久| 欧美最顶级丰满的aⅴ艳星| 欧美午夜宅男影院在线观看| 亚洲第一男人av| 亚洲国产女人aaa毛片在线| 久久精品欧美视频| 日韩av综合网站| 亚洲精品国产品国语在线| 精品中文字幕在线| 亚洲人成网在线播放| 欧美性理论片在线观看片免费| 日韩欧美精品免费在线| 亚洲www永久成人夜色| 久久精品国产一区二区三区| 国产99视频精品免视看7| 最近2019免费中文字幕视频三| 一区二区三欧美| 国产精品普通话| 午夜精品久久久久久久99热浪潮| 国产91精品久久久久| 欧美成人高清视频| 久久久国产精彩视频美女艺术照福利| 中文字幕日韩精品在线观看| 最好看的2019年中文视频| 97久久精品视频| 欧美激情一区二区三区久久久| 亚洲黄色有码视频| **欧美日韩vr在线| 91精品国产综合久久久久久久久| 国产午夜精品美女视频明星a级| 亚洲第一中文字幕在线观看| 国产女人18毛片水18精品| 性色av香蕉一区二区| 久久久成人的性感天堂| 亚洲精品国产美女| 91国偷自产一区二区三区的观看方式| 国产精品激情av电影在线观看| 成人久久一区二区| 成人欧美一区二区三区在线湿哒哒| 69国产精品成人在线播放| 色悠悠国产精品| 国模精品视频一区二区三区| 国产综合在线看| 欧美成人免费观看| 欧美限制级电影在线观看| 国产一区二区三区视频| 亚洲美女av网站| 欧美诱惑福利视频| 国产欧美一区二区三区在线看| 国产视频亚洲精品| 亚洲欧美日韩视频一区| 亚洲小视频在线观看| 精品国产成人av| 69久久夜色精品国产7777| 久久精品成人欧美大片古装| 亚洲欧美综合图区| 欧美精品久久久久| 成人激情视频在线播放| 啪一啪鲁一鲁2019在线视频| 欧美另类极品videosbestfree| 欧美黄色成人网| 精品伊人久久97| 欧美日韩免费区域视频在线观看| 国产成+人+综合+亚洲欧洲| 国产精品久久久av久久久| 久久久久久久久久久免费| 亚洲三级 欧美三级| 国产成人精品一区二区在线| 亚洲天堂第二页| 亚洲视频在线免费观看| 国产精品丝袜久久久久久不卡| 日韩高清av一区二区三区| 欧美性极品xxxx做受| 国产精品十八以下禁看| 精品国产一区二区在线| 亚洲男人的天堂在线| 亚洲国产欧美一区二区三区久久| 日韩一区二区三区国产| 精品国产一区二区三区久久狼5月| 亚洲一区二区三区xxx视频| 中文字幕日韩欧美在线| 久久久久久91| 久久成人国产精品| 日本国产高清不卡| 欧美激情精品久久久久久黑人| 国产日韩欧美综合| 中文字幕日韩精品在线| 亚洲精品久久久久久久久久久久| 亚洲va男人天堂| 国产日韩精品入口| 亚洲免费人成在线视频观看| 亚洲成人久久电影| 日韩亚洲精品电影| 久久久免费观看视频| 91精品国产综合久久久久久久久| 亚洲综合精品伊人久久| 色综合天天综合网国产成人网| 97视频在线观看播放| 欧美多人爱爱视频网站| 亚洲日本中文字幕| 成人女保姆的销魂服务| 日韩成人激情视频| 久久精品国产69国产精品亚洲| 久久精品中文字幕一区| 欧美亚洲国产另类| 国产欧美一区二区三区视频| 在线观看国产精品91| 欧美亚洲日本黄色| 亚洲欧美一区二区三区四区| 日韩免费视频在线观看| 久久精品99久久久久久久久| 国产一区二区视频在线观看| 国产欧美日韩高清| 91极品视频在线| 亚洲精品中文字幕女同| 成人美女av在线直播| 日韩中文在线视频| 久久国产精品久久久久久| 亚洲毛片在线观看| 高潮白浆女日韩av免费看| 亚洲美女精品久久| 欧美床上激情在线观看| 最近免费中文字幕视频2019| 国语自产精品视频在线看抢先版图片| 免费97视频在线精品国自产拍| 夜夜嗨av一区二区三区四区| 亚洲欧美日本精品| 色诱女教师一区二区三区| 日韩美女av在线免费观看| 亚洲成人精品在线| 91美女福利视频高清| 亚洲第一福利视频| 成人精品在线观看| 夜夜嗨av一区二区三区四区| 日韩美女在线观看| 亚洲系列中文字幕| 超碰97人人做人人爱少妇| 色樱桃影院亚洲精品影院| 57pao国产精品一区| 国产精品视频在线播放| 日韩高清av一区二区三区| 国产精品主播视频| 中文字幕少妇一区二区三区| 国产精品精品久久久久久| 欧美激情一级欧美精品| 欧洲日韩成人av| 久久久久久国产精品美女| 欧美亚洲另类制服自拍| 狠狠色香婷婷久久亚洲精品| 亚洲精品视频久久| 欧美性生交大片免费| 亚洲色图35p| 国产日韩欧美在线视频观看| 欧美性精品220| 日韩电影免费观看在线| 亚洲欧美精品在线| 成人欧美一区二区三区黑人| 欧美成人精品xxx| 日韩精品高清视频| 国产精品嫩草视频| www.日韩.com| 另类专区欧美制服同性| 国产精品成人aaaaa网站| 国产91精品视频在线观看|