再谈PHP错误与异常处理

  博客很久没有更新了,实在惭愧,最近在忙人生大事,哈哈!这段时间没有看什么新的东西,结合项目中遇到的PHP异常处理问题,我又从新梳理了以前模糊的概念,但愿对你们理解PHP异常处理有所帮助。php

  请必定要注意,没有特殊说明:本例 PHP Version < 7
  提及PHP异常处理,你们首先会想到try-catch,那好,咱们先看一段程序吧:有一个test.php文件,有一段简单的PHP程序,内容以下,而后命令行执行:php test.php
html

1 <?php
2     $num = 0;
3     try {
4         echo 1/$num;
5
6 } catch (Exception $e){ 7 echo $e->getMessage(); 8 } 9 ?>

  个人问题是:这段程序能正确的捕捉到除0的错误信息吗?
  若是你回答能,那你就把这篇文章看完吧!应该能学点东西。java

本文章分5个部分介绍个人异常处理的理解:数据库

1、异常与错误的概述服务器

2、ERROR的级别框架

3、PHP异常处理中的黑科技函数

4、巧妙的捕获错误和异常测试

5、自定义异常处理和异常嵌套网站

6、PHP7中的异常处理ui

1、异常与错误的概述
  PHP中什么是异常:
  程序在运行中出现不符合预期的状况,容许发生(你也不想让他出现不正常的状况)但他是一种不正常的状况,按照咱们的正常逻辑本不应出的错误,但仍然会出现的错误,属于逻辑和业务流程的错误,而不是编译或者语法上的错误。

  PHP中什么是错误:
  属于php脚本自身的问题,大部分状况是由错误的语法,服务器环境致使,使得编译器没法经过检查,甚至没法运行的状况。warning、notice都是错误,只是他们的级别不一样而已,而且错误是不能被try-catch捕获的。

  上面的说法是有前提条件的:
  在PHP中,由于在其余语言中就不能这样下结论了,也就是说异常和错误的说法在不一样的语言有不一样的说法。在PHP中任何自身的错误或者是非正常的代码都会当作错误对待,并不会以异常的形式抛出,可是也有一些状况会当作异常和错误同时抛出(听说是,我没有找到合适的例子)。也就是说,你想在数据库链接失败的时候自动捕获异常是行不通的,由于这就不是异常,是错误。可是在java中就不同了,他会把不少和预期不一致的行为当作异常来进行捕获。

  PHP异常处理很鸡肋?
  在上面的分析中咱们能够看出,PHP并不能主动的抛出异常,可是你能够手动抛出异常,这就很无语了,若是你知道哪里会出问题,你添加if else解决不就好了吗,为啥还要手动抛出异常,既然能手动抛出就证实这个不是异常,而是意料之中。以个人理解,这就是PHP异常处理鸡肋的地方(不必定对啊)。因此PHP的异常机制不是那么的完美,可是使用过框架的同窗都知道有这个状况:你在框架中直接写开头那段php“自动”捕获异常的代码是能够的,这是为何?看过源码的同窗都知道框架中都会涉及三个函数:register_shutdown_function,set_error_handler,set_exception_handler后面我会重点讲解着三个黑科技,经过这几个函数咱们能够实现PHP假自动捕获异常和错误。

2、ERROR的级别
  只有熟悉错误级别才能对错误捕捉有更好的认识。 ERROR有不一样的错误级别,我以前的一篇文章中有写到:http://www.cnblogs.com/zyf-zhaoyafei/p/3649434.html
  下面我再总结性的给出这几类错误级别:

 1     Fatal Error:致命错误(脚本终止运行)
 2         E_ERROR         // 致命的运行错误,错误没法恢复,暂停执行脚本
 3         E_CORE_ERROR    // PHP启动时初始化过程当中的致命错误
 4         E_COMPILE_ERROR // 编译时致命性错,就像由Zend脚本引擎生成了一个E_ERROR
 5         E_USER_ERROR    // 自定义错误消息。像用PHP函数trigger_error(错误类型设置为:E_USER_ERROR)
 6 
 7     Parse Error:编译时解析错误,语法错误(脚本终止运行)
 8         E_PARSE  //编译时的语法解析错误
 9 
10     Warning Error:警告错误(仅给出提示信息,脚本不终止运行)
11         E_WARNING         // 运行时警告 (非致命错误)。
12         E_CORE_WARNING    // PHP初始化启动过程当中发生的警告 (非致命错误) 。
13         E_COMPILE_WARNING // 编译警告
14         E_USER_WARNING    // 用户产生的警告信息
15 
16     Notice Error:通知错误(仅给出通知信息,脚本不终止运行)
17         E_NOTICE      // 运行时通知。表示脚本遇到可能会表现为错误的状况.
18         E_USER_NOTICE // 用户产生的通知信息。

  由此可知有5类是产生ERROR级别的错误,这种错误直接致使PHP程序退出。
  能够定义成:

1 ERROR = E_ERROR | E_CORE_ERROR |  E_COMPILE_ERROR | E_USER_ERROR | E_PARSE

3、PHP异常处理中的黑科技
  前面提到框架中是能够捕获全部的错误和异常的,之因此能实现应该是使用了黑科技,哈哈!其实也不是什么黑科技,主要是三个重要的函数:

  1:set_error_handler()
  看到这个名字估计就知道什么意思了,这个函数用于捕获错误,设置一个用户自定义的错误处理函数。

1 <?php
2     set_error_handler('zyferror');
3     function zyferror($type, $message, $file, $line)
4     {
5       var_dump('<b>set_error_handler: ' . $type . ':' . $message . ' in ' . $file . ' on ' . $line . ' line .</b><br />');
6     }
7 ?>

  当程序出现错误的时候自动调用此方法,不过须要注意一下两点:第一,若是存在该方法,相应的error_reporting()就不能在使用了。全部的错误都会交给自定义的函数处理。第二,此方法不能处理如下级别的错误:E_ERROR、 E_PARSE、 E_CORE_ERROR、 E_CORE_WARNING、 E_COMPILE_ERROR、 E_COMPILE_WARNING,set_error_handler() 函数所在文件中产生的E_STRICT,该函数只能捕获系统产生的一些Warning、Notice级别的错误。
  而且他有多种调用的方法:

1 <?php
2      // 直接传函数名 NonClassFunction
3      set_error_handler('function_name'); 
4 
5      // 传 class_name && function_name
6      set_error_handler(array('class_name', 'function_name')); 
7 ?>

   2:register_shutdown_function()
  捕获PHP的错误:Fatal Error、Parse Error等,这个方法是PHP脚本执行结束前最后一个调用的函数,好比脚本错误、die()、exit、异常、正常结束都会调用,多么牛逼的一个函数啊!经过这个函数就能够在脚本结束前判断此次执行是否有错误产生,这时就要借助于一个函数:error_get_last();这个函数能够拿到本次执行产生的全部错误。error_get_last();返回的信息:
  [type]           - 错误类型
  [message] - 错误消息
  [file]              - 发生错误所在的文件
  [line]             - 发生错误所在的行

1 <?php
2     register_shutdown_function('zyfshutdownfunc');
3     function zyfshutdownfunc()
4     {
5         if ($error = error_get_last()) {
6             var_dump('<b>register_shutdown_function: Type:' . $error['type'] . ' Msg: ' . $error['message'] . ' in ' . $error['file'] . ' on line ' . $error['line'] . '</b>');
7         }
8     }
9 ?>

   经过这种方法就能够巧妙的打印出程序结束前全部的错误信息。可是我在测试的时候我发现并非全部的错误终止后都会调用这个函数,能够看下面的一个测试文件,内容是:

 1 <?php
 2     register_shutdown_function('zyfshutdownfunc');
 3     function zyfshutdownfunc()
 4     {
 5         if ($error = error_get_last()) {
 6             var_dump('<b>register_shutdown_function: Type:' . $error['type'] . ' Msg: ' . $error['message'] . ' in ' . $error['file'] . ' on line ' . $error['line'] . '</b>');
 7         }
 8     }
 9     var_dump(23+-+); //此处语法错误
10 ?>

   本身能够试一下,你能够看到根本就不会触发zyfshutdownfunc()函数,其实这是一个语法错误,直接报了一个:

1 <?php
2     Parse error: syntax error, unexpected ')' in /www/mytest/exception/try-catch.php on line 71
3 ?>

  由此引出一个奇葩的问题:问什么不能触发,为何框架中是能够的?其实缘由很简单,只在parse-time出错时是不会调用本函数的。只有在run-time出错的时候,才会调用本函数,个人理解是语法检查器前没有执行register_shutdown_function()去把须要注册的函数放到调用的堆栈中,因此就根本不会运行。那框架中为何任何错误都能进入到register_shutdown_function()中呢,其实在框架中通常会有统一的入口index.php,而后每一个类库文件都会经过include ** 的方式加载到index.php中,至关与全部的程序都会在index.php中汇集,一样,你写的具备语法错误的文件也会被引入到入口文件中,这样的话,调用框架,执行index.php,index.php自己并无语法错误,也就不会产生parse-time错误,而是 include 文件出错了,是run-time的时候出错了,因此框架执行完以后就会触发register_shutdown_function();
  因此如今但是试一下这个写法,这样就会触发zyfshutdownfunc()回调了:

 1 a.php文件
 2 <?php
 3   // 模拟语法错误
 4   var_dump(23+-+);
 5 ?>
 6 
 7 b.php文件
 8 <?php
 9     register_shutdown_function('zyfshutdownfunc');
10     function zyfshutdownfunc()
11     {
12         if ($error = error_get_last()) {
13             var_dump('<b>register_shutdown_function: Type:' . $error['type'] . ' Msg: ' . $error['message'] . ' in ' . $error['file'] . ' on line ' . $error['line'] . '</b>');
14         }
15     }
16     require 'a.php';
17 ?>

   3:set_exception_handler()
  设置默认的异常处理程序,用在没有用try/catch块来捕获的异常,也就是说无论你抛出的异常有没有人捕获,若是没有人捕获就会进入到该方法中,而且在回调函数调用后异常会停止。看一下用法:

1 <?php
2     set_exception_handler('zyfexception');
3     function zyfexception($exception)
4     {
5         var_dump("<b>set_exception_handler: Exception: " . $exception->getMessage()  . '</b>');
6     }
7     throw new Exception("zyf exception");
8 ?>

 4、巧妙的捕获错误和异常
  1:把错误以异常的形式抛出(不能彻底抛出)
    由上面的讲解咱们知道,php中的错误是不能以异常的像是捕获的,可是咱们须要让他们抛出,已达到扩展 try-catch的影响范围,咱们前面讲到过set_error_handler() 方法,他是干吗用的,他是捕获错误的,因此咱们就能够借助他来吧错误捕获,而后再以异常的形式抛出,ok,试试下面的写法:

 1 <?php
 2     set_error_handler('zyferror');
 3     function zyferror($type, $message, $file, $line)
 4     {
 5         throw new \Exception($message . 'zyf错误当作异常');
 6     }
 7 
 8     $num = 0;
 9     try {
10         echo 1/$num;
11 
12     } catch (Exception $e){
13         echo $e->getMessage();
14     }
15 ?>

  好了,试一下,会打印出:

1 Division by zero zyf123

  流程:原本是除0错误,而后触发set_error_handler(),在set_error_handler()中至关与杀了个回马枪,再把错误信息以异常的形式抛出来,这样就能够实现错误以异常的形式抛出。你们要注意:这样作是有缺点的,会受到set_error_handler()函数捕获级别的限制。

     2:捕获全部的错误
        由set_error_handler()可知,他可以捕获一部分错误,不能捕获系统级E_ERROR、E_PARSE等错误,可是这部分能够由register_shutdown_function()捕获。因此二者结合能出现很好的功能。
        看下面的程序:

 1 a.php内容:
 2 <?
 3     // 模拟Fatal error错误
 4     //test();
 5 
 6     // 模拟用户产生ERROR错误
 7     //trigger_error('zyf-error', E_USER_ERROR);
 8 
 9     // 模拟语法错误
10     var_dump(23+-+);
11 
12     // 模拟Notice错误
13     //echo $f;
14 
15     // 模拟Warning错误
16     //echo '123';
17     //ob_flush();
18     //flush();
19     //header("Content-type:text/html;charset=gb2312");
20 ?>
21 b.php内容: 22 <? 23 error_reporting(0); 24 register_shutdown_function('zyfshutdownfunc'); 25 function zyfshutdownfunc() 26 { 27 if ($error = error_get_last()) { 28 var_dump('<b>register_shutdown_function: Type:' . $error['type'] . ' Msg: ' . $error['message'] . ' in ' . $error['file'] . ' on line ' . $error['line'] . '</b>'); 29 } 30 } 31 32 set_error_handler('zyferror'); 33 function zyferror($type, $message, $file, $line) 34 { 35 var_dump('<b>set_error_handler: ' . $type . ':' . $message . ' in ' . $file . ' on ' . $line . ' line .</b><br />'); 36 } 37 38 require 'a.php'; 39 ?>

   到此就能够解释开头的那个程序了吧,test.php 若是是单文件执行是不能捕获到错误的,若是你在框架中执行就是能够的,固然你按照我上面介绍的来扩展也是能够的。

5、自定义异常处理和异常嵌套

  1:自定义异常处理

  在复杂的系统中,咱们每每须要本身捕获咱们须要特殊处理的异常,这些异常多是特殊状况下抛出的。因此咱们就本身定义一个异常捕获类,该类必须是 exception 类的一个扩展,该类继承了 PHP 的 exception 类的全部属性,而且咱们能够添加自定义的函数,使用的时候其实和以前的同样,大体写法以下:

 1 <?php
 2     class zyfException extends Exception
 3     {
 4         public function errorzyfMessage()
 5         {
 6             return 'Error line ' . $this->getLine().' in ' . $this->getFile()
 7                 .': <b>' . $this->getMessage() . '</b> Must in (0 - 60)';
 8         }
 9     }
10 
11     $age = 10;
12     try {
13         $age = intval($age);
14         if($age > 60) {
15             throw new zyfException($age);
16         }
17 
18     } catch (zyfException $e) {
19         echo $e->errorzyfMessage();
20 
21     }
22 ?>

  2:异常嵌套

  异常嵌套是比较常见的写法,在自定义的异常处理中,try 块中能够定义多个异常捕获,而后分层传递异常,理解和冒泡差很少,看下面的实现:

 1 <?php
 2     $age = 10;
 3     try {
 4         $age = intval($age);
 5         if($age > 60) {
 6             throw new zyfException($age);
 7         }
 8 
 9         if ($age <= 0) {
10             throw new Exception($age . ' must > 0');
11         }
12 
13     } catch (zyfException $e) {
14         echo $e->errorzyfMessage();
15 
16     } catch(Exception $e) {
17         echo $e->getMessage();
18     }
19 ?>

  固然也能够在catch中再抛出异常给上层:

 1 <?php
 2     $age = 100;
 3     try {
 4         try {
 5             $age = intval($age);
 6             if($age > 60) {
 7                 throw new Exception($age);
 8             }
 9 
10         } catch (Exception $e) {
11             throw new zyfException($age);
12 
13         }
14 
15     } catch (zyfException $e) {
16         echo $e->errorzyfMessage();
17     }
18 ?>

6、PHP7中的异常处理
  如今写PHP必须考虑版本状况,上面的写法在PHP7中大部分都能实现,可是也会有不一样点,在PHP7更新中有一条:更多的Error变为可捕获的Exception,如今的PHP7实现了一个全局的throwable接口,原来老的Exception和其中一部分Error实现了这个接口(interface),PHP7中更多的Error变为可捕获的Exception返回给捕捉器,这样其实和前面提到的扩展try-catch影响范围同样,可是若是不捕获则仍是按照Error对待,看下面两个:

 1 <?php
 2     try {
 3         test();
 4 
 5     } catch(Throwable $e) {
 6         echo $e->getMessage() . ' zyf';
 7     }
 8 
 9     try {
10         test();
11 
12     } catch(Error $e) {
13         echo $e->getMessage() . ' zyf';
14     }
15 ?>

 由于PHP7实现了throwable接口,那么就可使用第一个这种方式来捕获异常。又由于部分Error实现了接口,而且更多的Error变为可捕获的Exception,那么就可使用第二种方式来捕获异常。下面是在网上找的PHP7的异常层次树:
Throwable
  Exception 异常
    ...
  Error 错误
    ArithmeticError 算数错误
      DivisionByZeroError 除数为0的错误
    AssertionError 声明错误
    ParseError 解析错误
    TypeError 类型错误

 就写到这吧,写得手疼,关于错误和异常处理的大体就写这么多,有什么错误请在评论中给出,多谢你们。

注意:
一、本博客同步更新到个人我的网站:http://www.zhaoyafei.cn
二、本文属原创内容,为了尊重他人劳动,转载请注明本文地址:

http://www.cnblogs.com/zyf-zhaoyafei/p/6928149.html 

相关文章
相关标签/搜索