error
error级别
fatal error:致命错误(脚本终止运行)
e_error 致命的运行时的致命错误,终止程序执行
e_core_error php 启动时的致命错误
e_compile_error php编译时的致命错误
e_user_error 用户产生的致命错误
parse error:编译时的解析错误(脚本终止运行)
parse error 编译时的语法解析错误
warning error:警告错误(仅给出提示信息,但是脚本不会终止运行。)
e_warning 运行时警告 (非致命错误)。
e_core_warning php初始化启动过程中发生的警告 (非致命错误) 。
e_compile_warning 编译警告
e_user_warning 用户产生的警告信息
notice error:通知错误(仅给出通知信息,但是脚本不会终止运行。)
e_notice 运行时通知。表示脚本遇到可能会表现为错误的情况.
e_user_notice 用户产生的通知信息。
set_error_handler()捕获错误【有局限】
函数说明
set_error_handler($callback);//设置一个用户的函数(error_handler)来处理脚本中出现的错误。
函数的局限性
以下级别的错误不能由用户定义的函数来处理: e_error、 e_parse、 e_core_error、 e_core_warning、 e_compile_error、 e_compile_warning,和在 调用 set_error_handler() 函数所在文件中产生的大多数 e_strict。
也就是:set_error_handler($callback)只能捕获系统产生的一些warning、notice级别的error。
使用方法
输出结果:
errno:8errstr:undefined variable: testerrfile:/users/shuchao/desktop/handler.phperrline:13
如何捕获php的fatal error、parse error等
需求描述
获取php的fatal error,比如记录到log里面,利于我们分析线上问题,可以做线上服务的监控。
两个函数
register_shutdown_function()
register_shutdown_function($callback)
register_shutdown_function(),就把你要注册进去的function放进【假装是队列吧】,等到脚本正常退出或显示调用exit时,再把注册进去的function拉出来执行.
register_shutdown_function()调用的3种情况:
脚本正常退出时;
在脚本运行(run-time not parse-time)出错退出时;
用户调用exit方法退出时。
error_get_last()
error_get_last();//函数获取最后发生的错误。
该函数以数组的形式返回最后发生的错误。
返回的数组包含 4 个键和值:
[type] - 错误类型
[message] - 错误消息
[file] - 发生错误所在的文件
[line] - 发生错误所在的行
使用方法
强烈注意
在parse-time出错的时候,是不会调用register_shutdown_function()函数的。只有在run-time出错的时候,才会调用register_shutdown_function()。
下面我们举例说明:
no.1
error_handler.php
执行结果如下:
fatal error: cannot redeclare test() (previously declared in /users/shuchao/desktop/error_handler.php:6) in/users/shuchao/desktop/error_handler.php on line 7
原因分析
在执行error_handler.php的时候,由于重复定义了两个函数test(),在php的parse-time就出错了(不是run-time),所以不能回调register_shutdown_function()中的函数。
no.2
error_handler.php
执行结果如下:
fatal error: cannot redeclare test() (previously declared in /users/shuchao/desktop/error_handler.php:9) in /users/shuchao/desktop/error_handler.php on line 7yeah,it's worked!%
原因分析
我们看到,上面回调了register_shutdown_function().
因为我们加了一个if()判断,if()里面的test()方法,相当于一个闭包,与外面的test()名称不冲突。
也就是,上面的代码在parse-time没有出错,而是在run-time的时候出错了,所以我们能够获取到fatal error。
no.3
error_handler.php
执行 test_error.php的结果如下
fatal error: cannot redeclare test() (previously declared in /users/shuchao/desktop/test_error.php:3) in/users/shuchao/desktop/test_error.php on line 4
原因分析
当我们在运行test_error.php的时候,因为redeclare了两个test()方法,所以php的语法解析器在parse-time的时候就出错了。 所以不能回调register_shutdown_function()中的方法,不能catch住这个fatal error。
no.4
error_handler.php
执行 include_all.php的结果如下
fatal error: cannot redeclare test() (previously declared in /users/shuchao/desktop/include_all.php:2) in /users/shuchao/desktop/include_all.php on line 3yeah,it's worked!%
结果分析
上面我们捕获了fatal_error.因为在运行include_all.php的时候,include_all.php本身语法并没有出错,也就是在parse-time的时候并没有出错,而是include的文件出错了,也就是在run-time的时候出错了,这个时候是能回调register_shutdown_function()中的函数的。
强烈建议:如果我们要使用register_shutdown_function进行错误捕捉,使用no.4,最后一种方法,可以确保错误都能捕捉到。
更优美的写法·获取所有错误
set_error_handler()与register_shutdown_function()、error_get_last()的结合使用
如何catch一个未捕获的exception
场景描述
假设程序中的有些地方直接throw了异常,没有进行catch。我们现在想要不管在程序的任何一个地方throw异常,即便在throw的地方没有被catch,我们也要能catch住,如何做到呢?
一个函数:set_exception_handler()
//设置默认的异常处理程序,用于没有用 try/catch 块来捕获的异常。 在 exception_handler 调用后异常会中止。
set_exception_handler()
使用示例
1、exception_handler.php
getmessage();}
2、test_exception.php
现在我们运行 test_exception.php,结果如下:
i am exception //证明我们throw的exception被捕获了