欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

PHP各種異常和錯(cuò)誤的攔截方法及發(fā)生致命錯(cuò)誤時(shí)進(jìn)行報(bào)警

 更新時(shí)間:2016年01月19日 13:41:36   作者:八面碰壁居士  
在日常開發(fā)中,大多數(shù)人的做法是在開發(fā)環(huán)境時(shí)開啟調(diào)試模式,在產(chǎn)品環(huán)境關(guān)閉調(diào)試模式。在開發(fā)的時(shí)候可以查看各種錯(cuò)誤、異常,但是在線上就把錯(cuò)誤顯示的關(guān)閉

在日常開發(fā)中,大多數(shù)人的做法是在開發(fā)環(huán)境時(shí)開啟調(diào)試模式,在產(chǎn)品環(huán)境關(guān)閉調(diào)試模式。在開發(fā)的時(shí)候可以查看各種錯(cuò)誤、異常,但是在線上就把錯(cuò)誤顯示的關(guān)閉。

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

但是你有沒(méi)有遇到這種情況,線下好好的,一上線卻運(yùn)行不起來(lái)也找不到原因...

一個(gè)腳本,跑了好長(zhǎng)一段時(shí)間,一直沒(méi)有問(wèn)題,有一天突然中斷了,然后了也沒(méi)有任何記錄都不造啥原因...

線上一個(gè)付款,別人明明付了款,但是我們卻沒(méi)有記錄到,自己親自去實(shí)驗(yàn),卻是好的...

種種以上,都是因?yàn)榇蠹谊P(guān)閉了錯(cuò)誤信息,并且未將錯(cuò)誤、異常記錄到日志,導(dǎo)致那些隨機(jī)發(fā)生的錯(cuò)誤很難追蹤。這樣矛盾就來(lái)了,即不要顯示錯(cuò)誤,又要追蹤錯(cuò)誤,這如何實(shí)現(xiàn)了?

以上問(wèn)題都可以通過(guò)PHP的錯(cuò)誤、異常機(jī)制及其內(nèi)建函數(shù)'set_exception_handler','set_error_handler','register_shutdown_function' 來(lái)實(shí)現(xiàn)

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

'set_error_handler' 函數(shù)可以攔截各種錯(cuò)誤,然后交給用戶自定義的方式進(jìn)行處理

'register_shutdown_function' 函數(shù)是在PHP腳本結(jié)束時(shí)調(diào)用的函數(shù),配合'error_get_last'可以獲取最后的致命性錯(cuò)誤

這個(gè)思路大體就是把錯(cuò)誤、異常、致命性錯(cuò)誤攔截下來(lái),交給我們自定義的方法進(jìn)行處理,我們辨別這些錯(cuò)誤、異常是否致命,如果是則記錄的數(shù)據(jù)庫(kù)或者文件系統(tǒng),然后使用腳本不停的掃描這些日志,發(fā)現(xiàn)嚴(yán)重錯(cuò)誤立即發(fā)送郵件或發(fā)送短信進(jìn)行報(bào)警

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

/**
 * 文件名稱:baseErrorHandler.class.php
 * 摘 要:錯(cuò)誤攔截器父類
 */
require 'errorHandlerException.class.php';//異常類
class errorHandler
{
 public $argvs = array();
 public  $memoryReserveSize = 262144;//備用內(nèi)存大小
 private $_memoryReserve;//備用內(nèi)存
 /**
  * 方  法:注冊(cè)自定義錯(cuò)誤、異常攔截器
  * 參  數(shù):void
  * 返  回:void
  */
 public function register()
 {
  ini_set('display_errors', 0);
  set_exception_handler(array($this, 'handleException'));//截獲未捕獲的異常
  set_error_handler(array($this, 'handleError'));//截獲各種錯(cuò)誤 此處切不可掉換位置
  //留下備用內(nèi)存 供后面攔截致命錯(cuò)誤使用
  $this->memoryReserveSize > 0 && $this->_memoryReserve = str_repeat('x', $this->memoryReserveSize);
  register_shutdown_function(array($this, 'handleFatalError'));//截獲致命性錯(cuò)誤
 }
 /**
  * 方  法:取消自定義錯(cuò)誤、異常攔截器
  * 參  數(shù):void
  * 返  回:void
  */
 public function unregister()
 {
  restore_error_handler();
  restore_exception_handler();
 }
 /**
  * 方  法:處理截獲的未捕獲的異常
  * 參  數(shù):Exception $exception
  * 返  回:void
  */
 public function handleException($exception)
 {
  $this->unregister();
  try
  {
   $this->logException($exception);
   exit(1);
  }
  catch(Exception $e)
  {
   exit(1);
  }
 }
 /**
  * 方  法:處理截獲的錯(cuò)誤
  * 參  數(shù):int  $code 錯(cuò)誤代碼
  * 參  數(shù):string $message 錯(cuò)誤信息
  * 參  數(shù):string $file 錯(cuò)誤文件
  * 參  數(shù):int  $line 錯(cuò)誤的行數(shù)
  * 返  回:boolean
  */
 public function handleError($code, $message, $file, $line)
 {
  //該處思想是將錯(cuò)誤變成異常拋出 統(tǒng)一交給異常處理函數(shù)進(jìn)行處理
  if((error_reporting() & $code) && !in_array($code, array(E_NOTICE, E_WARNING, E_USER_NOTICE, E_USER_WARNING, E_DEPRECATED)))
  {//此處只記錄嚴(yán)重的錯(cuò)誤 對(duì)于各種WARNING NOTICE不作處理
   $exception = new errorHandlerException($message, $code, $code, $file, $line);
   $trace = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS);
   array_shift($trace);//trace的第一個(gè)元素為當(dāng)前對(duì)象 移除
   foreach($trace as $frame) 
   {
    if($frame['function'] == '__toString') 
    {//如果錯(cuò)誤出現(xiàn)在 __toString 方法中 不拋出任何異常
     $this->handleException($exception);
     exit(1);
    }
   }
   throw $exception;
  }
  return false;
 }
 /**
  * 方  法:截獲致命性錯(cuò)誤
  * 參  數(shù):void
  * 返  回:void
  */
 public function handleFatalError()
 {
  unset($this->_memoryReserve);//釋放內(nèi)存供下面處理程序使用
  $error = error_get_last();//最后一條錯(cuò)誤信息
  if(errorHandlerException::isFatalError($error))
  {//如果是致命錯(cuò)誤進(jìn)行處理
   $exception = new errorHandlerException($error['message'], $error['type'], $error['type'], $error['file'], $error['line']);
   $this->logException($exception);
   exit(1);
  }
 }
 /**
  * 方  法:獲取服務(wù)器IP
  * 參  數(shù):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; 
 }
 /**
  * 方  法:獲取當(dāng)前URI信息
  * 參  數(shù):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;
 }
 /**
  * 方  法:記錄異常信息
  * 參  數(shù):errorHandlerException $e 錯(cuò)誤異常
  * 返  回: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()),//這里為各種錯(cuò)誤定義一個(gè)編號(hào)以便查找
      '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);
 }
 /**
  * 方  法:記錄異常信息
  * 參  數(shù):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)
 {
  /*這里去實(shí)現(xiàn)如何將錯(cuò)誤信息記錄到日志*/
 }
}

上述代碼中,有個(gè)'errorHandlerException'類,該類放在文件'errorHandlerException.class.php'中,該類用于將錯(cuò)誤轉(zhuǎn)換為異常,以便記錄錯(cuò)誤發(fā)生的文件、行號(hào)、錯(cuò)誤代碼、錯(cuò)誤信息等信息,同時(shí)其方法'isFatalError'用于辨別該錯(cuò)誤是否是致命性錯(cuò)誤。這里我們?yōu)榱朔奖愎芾?,將錯(cuò)誤進(jìn)行編號(hào)并命名。該類的代碼如下

/**
 * 文件名稱:errorHandlerException.class.php
 * 摘 要:自定義錯(cuò)誤異常類 該類繼承至PHP內(nèi)置的錯(cuò)誤異常類
 */
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',
         );
 /**
  * 方  法:構(gòu)造函數(shù)
  * 摘  要:相關(guān)知識(shí)請(qǐng)查看 http://php.net/manual/en/errorexception.construct.php
  * 
  * 參  數(shù):string  $message  異常信息(可選)
  *    int   $code   異常代碼(可選)
  *    int   $severity
  *    string  $filename  異常文件(可選)
  *    int   $line   異常的行數(shù)(可選)
  *   Exception $previous 上一個(gè)異常(可選)
  *
  * 返  回:void
  */
 public function __construct($message = '', $code = 0, $severity = 1, $filename = __FILE__, $line = __LINE__, Exception $previous = null)
 {
  parent::__construct($message, $code, $severity, $filename, $line, $previous);
 }
 /**
  * 方  法:是否是致命性錯(cuò)誤
  * 參  數(shù):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);
 }
 /**
  * 方  法:根據(jù)原始的錯(cuò)誤代碼得到本地的錯(cuò)誤代碼
  * 參  數(shù):int $code
  * 返  回:int $localCode
  */
 public static function getLocalCode($code)
 {
  return isset(self::$localCode[$code]) ? self::$localCode[$code] : self::$localCode[4016];
 }
 /**
  * 方  法:根據(jù)原始的錯(cuò)誤代碼獲取用戶友好型名稱
  * 參  數(shù):int 
  * 返  回:string $name
  */
 public static function getName($code)
 {
  return isset(self::$localName[$code]) ? self::$localName[$code] : self::$localName[4016];
 }

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

然后我們定義一個(gè)文件,用于實(shí)例化以上類,捕獲各種錯(cuò)誤、異常,該文件命名為'registerErrorHandler.php', 內(nèi)如如下

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

剩下的就是需要你在你的入口文件引入該文件,定義調(diào)試模式,然后實(shí)現(xiàn)你自己記錄錯(cuò)誤的方法即可

需要注意的是,有些錯(cuò)誤在你進(jìn)行注冊(cè)之前已經(jīng)發(fā)生并且導(dǎo)致腳本中斷是無(wú)法記錄下來(lái)的,因?yàn)榇藭r(shí)'registerErrorHandler::register()' 尚未執(zhí)行已經(jīng)中斷了

還有就是'set_error_handler'這個(gè)函數(shù)不能捕獲下面類型的錯(cuò)誤 E_ERROR、 E_PARSE、 E_CORE_ERROR、 E_CORE_WARNING、E_COMPILE_ERROR、 E_COMPILE_WARNING, 這個(gè)可以在官方文檔中看到,但是本處無(wú)妨,因?yàn)橐陨襄e(cuò)誤是解析、編譯錯(cuò)誤,這些都沒(méi)有通過(guò),你是不可能發(fā)布上線的

相關(guān)文章

  • thinkphp獲取欄目和文章當(dāng)前位置的方法

    thinkphp獲取欄目和文章當(dāng)前位置的方法

    這篇文章主要介紹了thinkphp獲取欄目和文章當(dāng)前位置的方法,通過(guò)一個(gè)自定義的遞歸函數(shù)讀取目錄來(lái)實(shí)現(xiàn)獲取欄目和文章當(dāng)前位置,是非常實(shí)用的技巧,需要的朋友可以參考下
    2014-10-10
  • PHPCMS V9 添加二級(jí)導(dǎo)航的思路詳解

    PHPCMS V9 添加二級(jí)導(dǎo)航的思路詳解

    這篇文章主要介紹了PHPCMS V9 添加二級(jí)導(dǎo)航所遇到些問(wèn)題,查詢導(dǎo)航欄的信息時(shí)返回的$r[arrchildid]與自己想象的不符,文檔上說(shuō)是返回子欄目id但是卻有些不同。下面小編通過(guò)分享本文給大家解答下
    2016-10-10
  • CodeIgniter配置之a(chǎn)utoload.php自動(dòng)加載用法分析

    CodeIgniter配置之a(chǎn)utoload.php自動(dòng)加載用法分析

    這篇文章主要介紹了CodeIgniter配置之a(chǎn)utoload.php自動(dòng)加載用法,結(jié)合實(shí)例形式較為詳細(xì)的分析了CodeIgniter自動(dòng)加載機(jī)制的原理與使用方法,需要的朋友可以參考下
    2016-01-01
  • phpstudy2018升級(jí)MySQL5.5為5.7教程(圖文)

    phpstudy2018升級(jí)MySQL5.5為5.7教程(圖文)

    這篇文章主要介紹了phpstudy2018升級(jí)MySQL5.5為5.7教程(圖文),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-10-10
  • Symfony2針對(duì)輸入時(shí)間進(jìn)行查詢的方法分析

    Symfony2針對(duì)輸入時(shí)間進(jìn)行查詢的方法分析

    這篇文章主要介紹了Symfony2針對(duì)輸入時(shí)間進(jìn)行查詢的方法,結(jié)合實(shí)例形式分析了Symfony2針對(duì)mysql及MongoDB的輸入時(shí)間進(jìn)行轉(zhuǎn)換與查詢的相關(guān)操作技巧,需要的朋友可以參考下
    2017-06-06
  • CI框架AR數(shù)據(jù)庫(kù)操作常用函數(shù)總結(jié)

    CI框架AR數(shù)據(jù)庫(kù)操作常用函數(shù)總結(jié)

    這篇文章主要介紹了CI框架AR數(shù)據(jù)庫(kù)操作常用函數(shù),結(jié)合實(shí)例形式總結(jié)分析了基于CI框架的數(shù)據(jù)庫(kù)增刪改查與緩存、結(jié)果集等相關(guān)操作函數(shù)與技巧,需要的朋友可以參考下
    2016-11-11
  • ThinkPHP CURD方法之limit方法詳解

    ThinkPHP CURD方法之limit方法詳解

    這篇文章主要介紹了ThinkPHP CURD方法的limit方法,需要的朋友可以參考下
    2014-06-06
  • php等比例縮放圖片及剪切圖片代碼分享

    php等比例縮放圖片及剪切圖片代碼分享

    這篇文章給大家分享的是使用php實(shí)現(xiàn)的等比例縮放圖片及剪切圖片的代碼,非常的簡(jiǎn)單實(shí)用,附上用法,有需要的小伙伴可以參考下。
    2016-02-02
  • PHP構(gòu)造二叉樹算法示例

    PHP構(gòu)造二叉樹算法示例

    本篇文章主要介紹了PHP構(gòu)造二叉樹算法示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-06-06
  • laravel使用組件實(shí)現(xiàn)微信網(wǎng)頁(yè)授權(quán)登入

    laravel使用組件實(shí)現(xiàn)微信網(wǎng)頁(yè)授權(quán)登入

    這篇文章主要介紹了laravel使用組件實(shí)現(xiàn)微信網(wǎng)頁(yè)授權(quán)登入,使用laravel組件 laravel-wechat調(diào)用,使用起來(lái)很方便,有需要的同學(xué)可以學(xué)習(xí)下
    2021-03-03

最新評(píng)論