之所以想着做错误和异常的自动处理是因为:

    用的公司自己的框架写API,没有异常和错误相关功能,

    而每次操作都进行try...catch,有点繁琐不说,感觉还很鸡肋,即使我catch到了,还是得写代码进行处理,哪怕封装了一个方法进行处理也还是繁琐,

    这种情况应该是程序自动进行处理,不该是这样弱智的人工try...catch,然后处理

    以及其他同事写代码时并不能严格的进行try...catch

    于是乎,想自己设计一个自动捕获并处理错误和异常的功能。

  发现了一篇深度好文:https://www.cnblogs.com/zyf-zhaoyafei/p/6928149.html

     理解了PHP中自带的  register_shutdown_function,set_error_handler,set_exception_handler这三个函数

  通过这三个函数我们可以实现PHP假 自动捕获异常和错误。PHP5也不支持ERROR的捕获,只能try...catch (Exception)

  三个函数必须理解,如果你想实现该功能,希望你可以先做个小demo体验下三个函数的神奇之处

  这三个函数需要你写在框架的生命周期比较靠前的地方,如果不知道写哪里,请写在入口文件中,或者在入口文件require一下(下面的源码可以直接require使用)

  因为如果写到其他文件中,而碰巧该文件有错误或者异常,那么这三个函数就不发生作用了,而入口文件我们都可以确保正确

  正文:

      首先介绍set_error_handler这个函数,这个函数用于捕获低级别的错误,该函数只能捕获系统产生的一些Warning、Notice级别的错误,并调用一个用户自定义的错误处理函数。

    比如在入口文件写入  set_error_handler('low_level_error'),并且我们建立了low_level_error 自定义方法来进行日志的记录 。

    当发生Warning、Notice级别的错误时,系统会自动调用low_level_error方法,四个参数"type"、 "message"、"file" 和 "line"也是系统传过来的,可以直接使用。

    注意:使用此函数后  error_reporting ( E_ALL )  无法提示warning和notice级别的警告, 如果本地开发,最好关闭此函数,在生产环境才使用;  使用 其它两个函数 时错误提示信息不受影响

    可以使用下面源码中的单个方法进行测试, 只需echo $str; exit; 就好

      第二个函数 register_shutdown_function, 这个函数与第一个函数搭配使用,可以捕捉到Fatal Error、Parse Error等高级别会让程序中止的错误,

    这个方法是PHP脚本执行结束前最后一个调用的函数,比如脚本错误、die()、exit、异常、正常结束都会调用。这个方法没有接受参数,所以需要借助error_get_last() ,error_get_last函数可以获取到

    系统最近发生的一个错误,同样包含  "type"、 "message"、"file" 和 "line"四个字段,需要注意的是 本函数捕捉到的 (2,8,32,128,512,1024,2048,8192)等错误都是警告级别的,自己酌情处理。

      第三个函数set_exception_handler, 这个函数可以捕获 没有用try/catch块来捕获的异常,并且在回调函数调用后异常会中止(这个我没有试,因为我捕捉异常后直接返回给前端错误code码了)。

  思路:通过set_error_handler 和 register_shutdown_function 两个函数可以捕捉到所有的错误,通过set_exception_handler 捕捉所有未处理的异常,这样程序中所有的错误和异常就被我们捕捉到了,

    接下来,我们可以通过自定义函数对它们进行操作,由于我的业务是直接向前端输出,因此只是记录相关日志后直接echo并退出,大家可以在自定义函数内自己决定做什么处理。

    还可以在两个自定义的错误处理函数中再抛出异常,由自定义的异常处理类统一处理,这也是一个不错的思路,大家可以尝试下。

  最后:

    下面是相关的代码,供大家参考和测试。

<?php
/**引入本文件后,PHP会自动处理错误和异常, php版本5.6, php7+版本需要小调整,运行时根据提示调整即可
* 本方法直接捕捉错误并记录日志,如有兴趣,可以捕捉到错误后再次抛出异常,将错误转为异常后,由set_exception_handler统一进行处理
* User: LiZheng 271648298@qq.com
* Date: 2019/3/21
*/
register_shutdown_function('high_level_error'); //使用了register_shutdown_function 后,当程序遇见一些致命错误时会自动调用函数 high_level_error
set_error_handler('low_level_error'); //使用了set_error_handler 后,当程序遇见Notice 和Warning错误时会自动调用函数 low_level_error
set_exception_handler('exception_log'); //使用了set_exception_handler后,当遇到所有的未捕获的异常时会自动调用函数 exception_log
/**
* 该方法只能捕获系统产生的一些Warning、Notice级别的错误
* @param $type
* @param $message
* @param $file
* @param $line
* User: LiZheng 271648298@qq.com
* Date: 2019/3/21
*/
function low_level_error($type, $message, $file, $line)
{
$time = date('Y-m-d H:i:s');
//拼接要记录成日志的信息
$str = "\n ".$time.' set_error_handler: ' . exception_self::get_type($type) . ':' . $message . ' in ' . $file . ' on ' . $line . ' line .';
$str = addslashes($str);
//记录日志 error_log是PHP自带函数,记录到指定位置,详情见官网
error_log($str,3,ROOT_DIR."/runtime/logs/php_error.log"); //ROOT_DIR为项目根目录
//对于notice和warning 级别的错误,只进行记录而不会终止程序(但在本地开发时希望提示,可以通过if语句判断)
//echo json_encode(array('code'=>“ERR_ERROR”, 'msg'=>$time, 'data'=>array()), true);exit;
} /**
* 捕捉一些致命错误
* User: LiZheng 271648298@qq.com
* Date: 2019/3/21
*/
function high_level_error()
{
//error_get_last() 获取最近一条发生的错误,包含"type"、 "message"、"file" 和 "line"
if ($error = error_get_last()) {
$time = date('Y-m-d H:i:s');
//拼接要记录成日志的信息
$str = "\n ".$time.' register_shutdown_function: Type:' . exception_self::get_type($error['type']) . ' Msg: ' . $error['message'] . ' in ' . $error['file'] . ' on line ' . $error['line'];
$str = addslashes($str);
//记录日志 error_log是PHP自带函数,记录到指定位置,还可以发送邮件以及其他操作
error_log($str,3,ROOT_DIR."/runtime/logs/php_error.log"); //ROOT_DIR为项目根目录
//直接输出接口的返回信息, 给前端一个错误码code, msg返回时间用于方便查找日志
//echo json_encode(array('code'=>“ERR_ERROR”, 'msg'=>$time, 'data'=>array()), true);exit;
if(in_array($error['type'], array(2,8,32,128,512,1024,2048,8192)))//本条件中的错误均为非致命错误, 8192 错误是弃用提示
{
// if(APP_ENV == 'LOCAL') //这里可以对本地开发环境时进行不同的处理
// {
// header('Content-Type: text/html; charset=utf-8');
// echo $str."\n";
// }
}else
{
// if(APP_ENV == 'LOCAL')
// {
// header('Content-Type: text/html; charset=utf-8');
// echo $str."\n";
// }
//直接输出接口的返回信息, 给前端一个错误码code, msg返回时间用于方便查找日志
echo json_encode(array('code'=>ERR_ERROR, 'msg'=>$time, 'data'=>array()), true);exit;
}
}
} /**
* 捕捉异常
* @param $exception
* User: LiZheng 271648298@qq.com
* Date: 2019/3/21
*/
function exception_log($exception)
{
$time = date('Y-m-d H:i:s');
//拼接要记录成日志的信息
$str = "\n ".$time." set_exception_handler: Exception: ". $exception->getMessage()." in ".$exception->getFile()." on line ". $exception->getLine();
$str = addslashes($str);
//记录日志
error_log($str,3,ROOT_DIR."/runtime/logs/php_error.log");
//输出信息
echo json_encode(array('code'=>“ERR_EXCEPTION”, 'msg'=>$time, 'data'=>array()), true);exit;
} /**
* Class error_handler
* Create on 2019/3/21 14:15
* User: LiZheng 271648298@qq.com
* Date: 2019/3/21
*/
class exception_self extends Exception
{
/**
* 构造函数
* @param string $message
* @param string $code
*/
public function __construct($message=null, $code=null)
{
parent::__construct($message, $code);
// self::log($this->__toString());
}
//PHP的错误级别
public static $type = array(
'1' => 'E_ERROR ',
'2' => 'E_WARNING ',
'4' => 'E_PARSE ',
'8' => 'E_NOTICE ',
'16' => 'E_CORE_ERROR ',
'32' => 'E_CORE_WARNING ',
'64' => 'E_COMPILE_ERROR ',
'128' => 'E_COMPILE_WARNING ',
'256' => 'E_USER_ERROR ',
'512' => 'E_USER_WARNING ',
'1024' => 'E_USER_NOTICE ',
'2048' => 'E_STRICT ',
'4096' => 'E_RECOVERABLE_ERROR ',
'8191' => 'E_ALL ',
); /**
* 通过数字 置换 对应的英文
* @param $key
* @return mixed
* User: LiZheng 271648298@qq.com
* Date: 2019/3/21
*/
public static function get_type($key)
{
if(isset(self::$type[$key]))
{
return self::$type[$key];
}else
{
return $key;
}
}
}

关于PHP自动捕捉处理错误和异常的尝试的更多相关文章

  1. C#不用union,而是有更好的方式实现 .net自定义错误页面实现 .net自定义错误页面实现升级篇 .net捕捉全局未处理异常的3种方式 一款很不错的FLASH时种插件 关于c#中委托使用小结 WEB网站常见受攻击方式及解决办法 判断URL是否存在 提升高并发量服务器性能解决思路

    C#不用union,而是有更好的方式实现   用过C/C++的人都知道有个union,特别好用,似乎char数组到short,int,float等的转换无所不能,也确实是能,并且用起来十分方便.那C# ...

  2. php错误及异常捕捉

    原文:php错误及异常捕捉 在实际开发中,错误及异常捕捉仅仅靠try{}catch()是远远不够的. 所以引用以下几中函数. a)   set_error_handler 一般用于捕捉  E_NOTI ...

  3. PHP错误与异常

    请一定要注意,没有特殊说明:本例 PHP Version < 7 说起PHP异常处理,大家首先会想到try-catch,那好,我们先看一段程序吧:有一个test.php文件,有一段简单的PHP程 ...

  4. Java错误和异常

    Java 异常结构图 在 Java 中,所有的异常都有一个共同的祖先 Throwable(可抛出).Throwable 指定代码中可用异常传播机制通过 Java 应用程序传输的任何问题的共性.  Th ...

  5. php中的错误和异常

    总结: php错误不会抛出异常,因此不能被catch,但会根据配置写入日志文件或者输出到浏览器,所以可以通过日志文件查看错误 php异常都必须自己抛出,并通过catch捕捉.SQL语句执行的错误好像可 ...

  6. Python学习笔记五:错误与异常

    一:常见异常与错误 BaseException 所有异常的基类SystemExit 解释器请求退出KeyboardInterrupt 用户中断执行(通常是输入^C)Exception 常规错误的基类S ...

  7. React中如何优雅的捕捉事件错误

    React中如何优雅的捕捉事件错误 前话 人无完人,所以代码总会出错,出错并不可怕,关键是怎么处理. 我就想问问大家react的错误怎么捕捉呢? 这个时候: 小白:怎么处理? 小白+: ErrorBo ...

  8. python学习笔记014——错误和异常

    Python有两种错误很容易辨认:语法错误和异常. 1 什么是语法错误 Python 的语法错误或者称之为解析错,是初学者经常碰到的,如下实例 if i>4 print("if语句输出 ...

  9. 2015/9/10 Python基础(11):错误和异常

    程序在执行的过程中会产生异常,出现错误在以前的一个时期是致命的,后来随着程序的发展,使得一些错误的处理方式是柔和的,发生错误会产生一些错误的诊断信息和一些模糊的提示.帮助我们来处理异常.今天将学习Py ...

随机推荐

  1. 学JAVA第五天,今天困得要死

    好不容易坚持到第五天了,继续继续!!! 今天老师没有讲JAVA的for循环,倒是讲了HTML的相关内容: 讲了JAVA代码怎么在HTML中运行. 只要在HTML加入这个 background-colo ...

  2. Java开发笔记(五十七)因抽象方法而产生的抽象类

    前面介绍了类的常见用法,令人感叹面向对象的强大,几乎日常生活中的所有事物,都可以抽象成Java的基类及其子类.然而抽象操作也有副作用,就是某个抽象而来的行为可能是不确定的,比如半夜鸡叫,如果是公鸡则必 ...

  3. 20190325-HTML框架、audio标签、vedio标签、source标签、HTML表单

    目录 1.HTML框架 frameset:框架标记 frame:框架内文件 iframe:内嵌框架 2.audio标签 src:URL(可以用source标签替代) autoplay:自动播放 pre ...

  4. Django的urls.py加载静态资源图片,TypeError: view must be a callable or a list/tuple in the case of include().

    Django的urls.py加载静态资源图片,TypeError: view must be a callable or a list/tuple in the case of include(). ...

  5. C#获得指定目录床架时间、更新时间和最后访问时间等信息的代码

    将做工程过程常用的内容片段备份一次,下面的内容内容是关于C#获得指定目录床架时间.更新时间和最后访问时间等信息的内容,希望能对小伙伴们也有用. using System;using System.IO ...

  6. 共创力咨询推出《静态代码分析(PCLint)高级实务培训》课程!

    [课程背景] C/C++语言的语法非常灵活性,尤其是指针及内存使用,这种灵活性使代码效率比较高,但同时也使得代码编写具有较大的随意性,另外C/C++编译器不进行强制类型检查,也不对数据边界和有效性进行 ...

  7. 初使用maven遇到各种问题记录

    Cannot change version of project facet Dynamic Web Module to 2.5? 解决办法:将web.xml配置文件中的<web-app ver ...

  8. Git命令备忘

    最近在用Git,查了点相关资料,逻辑依然不太明了,先整理一部分备忘,以后补充 一.本地Git与Github/码云的关联 1. 设置本地用户名,邮箱 git config --global user.n ...

  9. 前后端分离djangorestframework—— 接入第三方的验证码平台

    关于验证码部分,在我这篇文章里说的挺详细的了:Python高级应用(3)—— 为你的项目添加验证码 这里还是再给一个前后端分离的实例,因为极验官网给的是用session作为验证的,而我们做前后端分离的 ...

  10. Django 数据流程图

    根据学习Django并且通过几个作业,发现Django制作网站的数据流程有些比较难懂,所以制作一个数据流程图,帮助自己理解,也希望对正学习的人有所帮助! 别的不多说,上美图: