当前位置:首页 > PHP教程 > php高级应用 > 列表

PHP各种异常和错误的拦截方法及发生致命错误时进行报警

发布:smiling 来源: PHP粉丝网  添加日期:2021-07-06 10:23:21 浏览: 评论:0 

在日常开发中,大多数人的做法是在开发环境时开启调试模式,在产品环境关闭调试模式。在开发的时候可以查看各种错误、异常,但是在线上就把错误显示的关闭。

上面的情形看似很科学,有人解释为这样很安全,别人看不到错误,以免泄露重要信息...

但是你有没有遇到这种情况,线下好好的,一上线却运行不起来也找不到原因...

一个脚本,跑了好长一段时间,一直没有问题,有一天突然中断了,然后了也没有任何记录都不造啥原因...

线上一个付款,别人明明付了款,但是我们却没有记录到,自己亲自去实验,却是好的...

种种以上,都是因为大家关闭了错误信息,并且未将错误、异常记录到日志,导致那些随机发生的错误很难追踪。这样矛盾就来了,即不要显示错误,又要追踪错误,这如何实现了?

以上问题都可以通过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'中,代码如下:

  1. /** 
  2.  * 文件名称:baseErrorHandler.class.php 
  3.  * 摘 要:错误拦截器父类 
  4.  */ 
  5. require 'errorHandlerException.class.php';//异常类 
  6. class errorHandler 
  7.  public $argvs = array(); 
  8.  public  $memoryReserveSize = 262144;//备用内存大小 
  9.  private $_memoryReserve;//备用内存 
  10.  /** 
  11.   * 方  法:注册自定义错误、异常拦截器 
  12.   * 参  数:void 
  13.   * 返  回:void 
  14.   */ 
  15.  public function register() 
  16.  { 
  17.   ini_set('display_errors', 0); 
  18.   set_exception_handler(array($this'handleException'));//截获未捕获的异常 
  19.   set_error_handler(array($this'handleError'));//截获各种错误 此处切不可掉换位置 
  20.   //留下备用内存 供后面拦截致命错误使用 
  21.   $this->memoryReserveSize > 0 && $this->_memoryReserve = str_repeat('x'$this->memoryReserveSize); 
  22.   register_shutdown_function(array($this'handleFatalError'));//截获致命性错误 
  23.  } 
  24.  /** 
  25.   * 方  法:取消自定义错误、异常拦截器 
  26.   * 参  数:void 
  27.   * 返  回:void 
  28.   */ 
  29.  public function unregister() 
  30.  { 
  31.   restore_error_handler(); 
  32.   restore_exception_handler(); 
  33.  } 
  34.  /** 
  35.   * 方  法:处理截获的未捕获的异常 
  36.   * 参  数:Exception $exception 
  37.   * 返  回:void 
  38.   */ 
  39.  public function handleException($exception
  40.  { 
  41.   $this->unregister(); 
  42.   try 
  43.   { 
  44.    $this->logException($exception); 
  45.    exit(1); 
  46.   } 
  47.   catch(Exception $e
  48.   { 
  49.    exit(1); 
  50.   } 
  51.  } 
  52.  /** 
  53.   * 方  法:处理截获的错误 
  54.   * 参  数:int  $code 错误代码 
  55.   * 参  数:string $message 错误信息 
  56.   * 参  数:string $file 错误文件 
  57.   * 参  数:int  $line 错误的行数 
  58.   * 返  回:boolean 
  59.   */ 
  60.  public function handleError($code$message$file$line
  61.  { 
  62.   //该处思想是将错误变成异常抛出 统一交给异常处理函数进行处理 
  63.   if((error_reporting() & $code) && !in_array($codearray(E_NOTICE, E_WARNING, E_USER_NOTICE, E_USER_WARNING, E_DEPRECATED))) 
  64.   {//此处只记录严重的错误 对于各种WARNING NOTICE不作处理 
  65.    $exception = new errorHandlerException($message$code$code$file$line); 
  66.    $trace = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS); 
  67.    array_shift($trace);//trace的第一个元素为当前对象 移除 
  68.    foreach($trace as $frame)  
  69.    { 
  70.     if($frame['function'] == '__toString')  
  71.     {//如果错误出现在 __toString 方法中 不抛出任何异常 
  72.      $this->handleException($exception); 
  73.      exit(1); 
  74.     } 
  75.    } 
  76.    throw $exception
  77.   } 
  78.   return false; 
  79.  } 
  80.  /** 
  81.   * 方  法:截获致命性错误 
  82.   * 参  数:void 
  83.   * 返  回:void 
  84.   */ 
  85.  public function handleFatalError() 
  86.  { 
  87.   unset($this->_memoryReserve);//释放内存供下面处理程序使用 
  88.   $error = error_get_last();//最后一条错误信息 
  89.   if(errorHandlerException::isFatalError($error)) 
  90.   {//如果是致命错误进行处理 
  91.    $exception = new errorHandlerException($error['message'], $error['type'], $error['type'], $error['file'], $error['line']); 
  92.    $this->logException($exception); 
  93.    exit(1); 
  94.   } 
  95.  } 
  96.  /** 
  97.   * 方  法:获取服务器IP 
  98.   * 参  数:void 
  99.   * 返  回:string 
  100.   */ 
  101.  final public function getServerIp() 
  102.  { 
  103.   $serverIp = ''
  104.   if(isset($_SERVER['SERVER_ADDR'])) 
  105.   { 
  106.    $serverIp = $_SERVER['SERVER_ADDR']; 
  107.   } 
  108.   elseif(isset($_SERVER['LOCAL_ADDR'])) 
  109.   { 
  110.    $serverIp = $_SERVER['LOCAL_ADDR']; 
  111.   } 
  112.   elseif(isset($_SERVER['HOSTNAME'])) 
  113.   { 
  114.    $serverIp = gethostbyname($_SERVER['HOSTNAME']); 
  115.   } 
  116.   else 
  117.   { 
  118.    $serverIp = getenv('SERVER_ADDR'); 
  119.   }   
  120.   return $serverIp;  
  121.  } 
  122.  /** 
  123.   * 方  法:获取当前URI信息 
  124.   * 参  数:void 
  125.   * 返  回:string $url 
  126.   */ 
  127.  public function getCurrentUri() 
  128.  { 
  129.   $uri = ''
  130.   if($_SERVER ["REMOTE_ADDR"]) 
  131.   {//浏览器浏览模式 
  132.    $uri = 'http://' . $_SERVER['SERVER_NAME'] . $_SERVER['REQUEST_URI']; 
  133.   } 
  134.   else 
  135.   {//命令行模式 
  136.    $params = $this->argvs; 
  137.    $uri = $params[0]; 
  138.    array_shift($params); 
  139.    for($i = 0, $len = count($params); $i < $len$i++) 
  140.    { 
  141.     $uri .= ' ' . $params[$i]; 
  142.    } 
  143.   } 
  144.   return $uri
  145.  } 
  146.  /** 
  147.   * 方  法:记录异常信息 
  148.   * 参  数:errorHandlerException $e 错误异常 
  149.   * 返  回:boolean 是否保存成功 
  150.   */ 
  151.  final public function logException($e
  152.  { 
  153.   $error = array
  154.       'add_time'  =>  time(), 
  155.       'title'  =>  errorHandlerException::getName($e->getCode()),//这里获取用户友好型名称 
  156.       'message'  =>  array(), 
  157.       'server_ip' =>  $this->getServerIp(), 
  158.       'code'   =>  errorHandlerException::getLocalCode($e->getCode()),//这里为各种错误定义一个编号以便查找 
  159.       'file'   => $e->getFile(), 
  160.       'line'   =>  $e->getLine(), 
  161.       'url'  => $this->getCurrentUri(), 
  162.      ); 
  163.   do 
  164.   { 
  165.    //$e->getFile() . ':' . $e->getLine() . ' ' . $e->getMessage() . '(' . $e->getCode() . ')' 
  166.    $message = (string)$e
  167.    $error['message'][] = $message
  168.   } while($e = $e->getPrevious()); 
  169.   $error['message'] = implode("\r\n"$error['message']); 
  170.   $this->logError($error); 
  171.  } 
  172.  /** 
  173.   * 方  法:记录异常信息 
  174.   * 参  数:array $error = array( 
  175.   *         'time' => int,  
  176.   *         'title' => 'string',  
  177.   *         'message' => 'string',  
  178.   *         'code' => int, 
  179.   *         'server_ip' => 'string' 
  180.   *          'file'  => 'string', 
  181.   *         'line' => int, 
  182.   *         'url' => 'string', 
  183.   *        ); 
  184.   * 返  回:boolean 是否保存成功 
  185.   */ 
  186.  public function logError($error
  187.  { 
  188.   /*这里去实现如何将错误信息记录到日志*/ 
  189.  } 

上述代码中,有个'errorHandlerException'类,该类放在文件'errorHandlerException.class.php'中,该类用于将错误转换为异常,以便记录错误发生的文件、行号、错误代码、错误信息等信息,同时其方法'isFatalError'用于辨别该错误是否是致命性错误。这里我们为了方便管理,将错误进行编号并命名。该类的代码如下:

  1. /** 
  2.  * 文件名称:errorHandlerException.class.php 
  3.  * 摘 要:自定义错误异常类 该类继承至PHP内置的错误异常类 
  4.  */ 
  5. class errorHandlerException extends ErrorException 
  6.  public static $localCode = array
  7.           E_COMPILE_ERROR => 4001, 
  8.           E_COMPILE_WARNING => 4002, 
  9.           E_CORE_ERROR => 4003, 
  10.           E_CORE_WARNING => 4004, 
  11.           E_DEPRECATED => 4005, 
  12.           E_ERROR => 4006, 
  13.           E_NOTICE => 4007, 
  14.           E_PARSE => 4008, 
  15.           E_RECOVERABLE_ERROR => 4009, 
  16.           E_STRICT => 4010, 
  17.           E_USER_DEPRECATED => 4011, 
  18.           E_USER_ERROR => 4012, 
  19.           E_USER_NOTICE => 4013, 
  20.           E_USER_WARNING => 4014, 
  21.           E_WARNING => 4015, 
  22.           4016 => 4016, 
  23.          ); 
  24.  public static $localName = array
  25.           E_COMPILE_ERROR => 'PHP Compile Error'
  26.           E_COMPILE_WARNING => 'PHP Compile Warning'
  27.           E_CORE_ERROR => 'PHP Core Error'
  28.           E_CORE_WARNING => 'PHP Core Warning'
  29.           E_DEPRECATED => 'PHP Deprecated Warning'
  30.           E_ERROR => 'PHP Fatal Error'
  31.           E_NOTICE => 'PHP Notice'
  32.           E_PARSE => 'PHP Parse Error'
  33.           E_RECOVERABLE_ERROR => 'PHP Recoverable Error'
  34.           E_STRICT => 'PHP Strict Warning'
  35.           E_USER_DEPRECATED => 'PHP User Deprecated Warning'
  36.           E_USER_ERROR => 'PHP User Error'
  37.           E_USER_NOTICE => 'PHP User Notice'
  38.           E_USER_WARNING => 'PHP User Warning'
  39.           E_WARNING => 'PHP Warning'
  40.           4016 => 'Customer`s Error'
  41.          ); 
  42.  /** 
  43.   * 方  法:构造函数 
  44.   * 摘  要:相关知识请查看 http://php.net/manual/en/errorexception.construct.php 
  45.   *  
  46.   * 参  数:string  $message  异常信息(可选) 
  47.   *    int   $code   异常代码(可选) 
  48.   *    int   $severity 
  49.   *    string  $filename  异常文件(可选) 
  50.   *    int   $line   异常的行数(可选) 
  51.   *   Exception $previous 上一个异常(可选) 
  52.   * 
  53.   * 返  回:void 
  54.   */ 
  55.  public function __construct($message = ''$code = 0, $severity = 1, $filename = __FILE__$line = __LINE__, Exception $previous = null) 
  56.  { 
  57.   parent::__construct($message$code$severity$filename$line$previous); 
  58.  } 
  59.  /** 
  60.   * 方  法:是否是致命性错误 
  61.   * 参  数:array $error 
  62.   * 返  回:boolean 
  63.   */ 
  64.  public static function isFatalError($error
  65.  { 
  66.   $fatalErrors = array
  67.         E_ERROR,  
  68.         E_PARSE,  
  69.         E_CORE_ERROR, 
  70.         E_CORE_WARNING,  
  71.         E_COMPILE_ERROR,  
  72.         E_COMPILE_WARNING 
  73.        ); 
  74.   return isset($error['type']) && in_array($error['type'], $fatalErrors); 
  75.  } 
  76.  /** 
  77.   * 方  法:根据原始的错误代码得到本地的错误代码 
  78.   * 参  数:int $code 
  79.   * 返  回:int $localCode 
  80.   */ 
  81.  public static function getLocalCode($code
  82.  { 
  83.   return isset(self::$localCode[$code]) ? self::$localCode[$code] : self::$localCode[4016]; 
  84.  } 
  85.  /** 
  86.   * 方  法:根据原始的错误代码获取用户友好型名称 
  87.   * 参  数:int  
  88.   * 返  回:string $name 
  89.   */ 
  90.  public static function getName($code
  91.  { 
  92.   return isset(self::$localName[$code]) ? self::$localName[$code] : self::$localName[4016]; 
  93.  } 

在错误拦截类中,需要用户自己定义实现错误记录的方法('logException'),这个地方需要注意,有些错误可能在一段时间内不断发生,因此只需记录一次即可,你可以使用错误代码、文件、行号、错误详情 生成一个MD5值用于记录该错误是否已经被记录,如果在规定时间内(一个小时)已经被记录过则不需要再进行记录

然后我们定义一个文件,用于实例化以上类,捕获各种错误、异常,该文件命名为'registerErrorHandler.php', 内如如下

  1. /* 
  2. * 使用方法介绍: 
  3. * 在入口处引入该文件即可,然后可以在该文件中定义调试模式常量'DEBUG_ERROR' 
  4. * 
  5. * <?php 
  6.  
  7. * require 'registerErrorHandler.php'; 
  8.  
  9. * ?> 
  10. */ 
  11. /** 
  12. * 调试错误模式: 
  13. * 0    =>   非调试模式,不显示异常、错误信息但记录异常、错误信息 
  14. * 1    =>   调试模式,显示异常、错误信息但不记录异常、错误信息 
  15. */ 
  16. define('DEBUG_ERROR', 0); 
  17. require 'errorHandler.class.php'
  18. class registerErrorHandler 
  19.  /** 
  20.   * 方  法:注册异常、错误拦截 
  21.   * 参  数:void 
  22.   * 返  回:void 
  23.   */ 
  24.  public static function register() 
  25.  { 
  26.   global $argv
  27.   if(DEBUG_ERROR) 
  28.   {//如果开启调试模式 
  29.    ini_set('display_errors', 1); 
  30.    return
  31.   } 
  32.   //如果不开启调试模式 
  33.   ini_set('error_reporting', -1); 
  34.   ini_set('display_errors', 0); 
  35.   $handler = new errorHandler(); 
  36.   $handler->argvs = $argv;//此处主要兼容命令行模式下获取参数 
  37.   $handler->register(); 
  38.  }  
  39. registerErrorHandler::register(); 

剩下的就是需要你在你的入口文件引入该文件,定义调试模式,然后实现你自己记录错误的方法即可。

需要注意的是,有些错误在你进行注册之前已经发生并且导致脚本中断是无法记录下来的,因为此时'registerErrorHandler::register()' 尚未执行已经中断了

还有就是'set_error_handler'这个函数不能捕获下面类型的错误 E_ERROR、 E_PARSE、 E_CORE_ERROR、 E_CORE_WARNING、E_COMPILE_ERROR、 E_COMPILE_WARNING, 这个可以在官方文档中看到,但是本处无妨,因为以上错误是解析、编译错误,这些都没有通过,你是不可能发布上线的。

Tags: PHP异常错误 PHP错误拦截

分享到: