PHP批斗大会之缺失的异常详解


Posted in PHP onJuly 09, 2019

故事的开始

这几天观察错误日志发现有一个数据反序列化的notice错误,实际情况我是从缓存中读取数据然后反序列化,因为反序列化失败,所以实际每次都是去数据库取的值。背后性能影响还是挺大的。

缺失的异常

刚开始写代码的时候一直不明白为什么要用异常,感觉if else就能搞定了,为什么还要多此一举,现在反而觉得 php 的异常太少。

对比两种序列化场景,一个是json,另一个是serialize。

json

在json encode/decode的时候,如果出现异常,可以通过json_last_error()来获取。

https://www.php.net/manual/en...

这样的设计只能说勉强够用,不太符合面向对象的套路。

serialize/unserialize

在使用自带的序列化和反序列化的时候,相比json的处理,则更加简单粗暴,没有函数能拿到最后的错误,只会通过自定义的error handler来接管,然后自己去做出一些相应的处理。

为什么要捕获异常

比如我的代码比较乱,有的 key 是 json 序列化,有的 key 是 serialize。我们可以将 key 分类。不能确保其他人配置的对应关系是对的,或者有的人忘记了,所以我需要用捕获异常的方式来兜底,这样我们的代码更加健壮一些。当unserialize失败之后,我们可以尝试去json_decode,而不是立即返回一个false,从而把请求传递到数据库。

代码演示

error_reporting(E_ALL);

$a = ["a" => 1];

class UnSerializeException extends ErrorException
{

}

set_error_handler(function ($severity, $message, $file, $line) {
  $info = explode(":", $message);

  if ($severity == E_NOTICE) {
    if ($info[0] == "unserialize()") {
      throw new UnSerializeException($message);
    }
    return true;
  } else {

    throw new ErrorException($message, 0, $severity, $file, $line);;
  }
});


try {
  $b = unserialize(json_encode($a));
} catch (ErrorException $exception) {
  var_dump(get_class($exception), $exception->getMessage(), $exception->getTraceAsString()); // 捕获到了
} finally {
  restore_error_handler();
}

try {
  $b = unserialize(json_encode($a));
} catch (ErrorException $exception) {
  var_dump(get_class($exception), $exception->getMessage(), $exception->getTraceAsString()); // 无法捕获
}

输出结果

string(20) "UnSerializeException"
string(43) "unserialize(): Error at offset 0 of 7 bytes"
string(181) "#0 [internal function]: {closure}(8, 'unserialize(): ...', '/Users/mengkang...', 34, Array)
#1 /Users/mengkang/PhpstormProjects/xxx/test.php(34): unserialize('{"a":1}')
#2 {main}"

Notice: unserialize(): Error at offset 0 of 7 bytes in /Users/mengkang/PhpstormProjects/xxx/test.php on line 42

后记

所以 php 代码的异常设计还是任重而道远的,而这些已经设定的“旧的规范”要推翻,需要“勇气”,毕竟会影响所有的使用者。

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对三水点靠木的支持。

PHP 相关文章推荐
PHP中的错误处理、异常处理机制分析
May 07 PHP
PHP系列学习之日期函数使用介绍
Aug 18 PHP
浅析Yii中使用RBAC的完全指南(用户角色权限控制)
Jun 20 PHP
解析Win7 XAMPP apache无法启动的问题
Jun 26 PHP
PHP上传文件时文件过大$_FILES为空的解决方法
Nov 26 PHP
对PHP新手的一些建议(PHP学习经验总结)
Aug 20 PHP
codeigniter显示所有脚本执行时间的方法
Mar 21 PHP
PHP中使用hidef扩展代替define提高性能
Apr 09 PHP
PHP+shell实现多线程的方法
Jul 01 PHP
PHP扩展mcrypt实现的AES加密功能示例
Jan 29 PHP
php+js实现的无刷新下载文件功能示例
Aug 23 PHP
PHP常量DIRECTORY_SEPARATOR原理及用法解析
Nov 10 PHP
PHP结合Redis+MySQL实现冷热数据交换应用案例详解
Jul 09 #PHP
PHP+Redis开发的书签案例实战详解
Jul 09 #PHP
使用composer命令加载vendor中的第三方类库 的方法
Jul 09 #PHP
Laravel+Intervention实现上传图片功能示例
Jul 09 #PHP
Laravel框架实现多个视图共享相同数据的方法详解
Jul 09 #PHP
Laravel5.1框架注册中间件的三种场景详解
Jul 09 #PHP
PHP使用 Pear 进行安装和卸载包的方法详解
Jul 08 #PHP
You might like
解析在apache里面给php写虚拟目录的详细方法
2013/06/24 PHP
php无限遍历文件夹示例分享
2014/03/04 PHP
PHP 7安装调试工具Xdebug扩展的方法教程
2017/06/17 PHP
PHP rsa加密解密算法原理解析
2020/12/09 PHP
Javascript 函数中的参数使用分析
2010/03/27 Javascript
jQuery 1.9.1源码分析系列(十三)之位置大小操作
2015/12/02 Javascript
关于cookie的初识和运用(js和jq)
2016/04/07 Javascript
jQuery实现动态文字搜索功能
2017/01/05 Javascript
实例解析js中try、catch、finally的执行规则
2017/02/24 Javascript
Nodejs实现文件上传的示例代码
2017/09/26 NodeJs
js中getter和setter用法实例分析
2018/08/14 Javascript
Vue 组件参数校验与非props特性的方法
2019/02/12 Javascript
详解vue更改头像功能实现
2019/04/28 Javascript
浅谈Vue2.4.0 $attrs与inheritAttrs的具体使用
2020/03/08 Javascript
OpenLayers3实现对地图的基本操作
2020/09/28 Javascript
[02:54]辉夜杯主赛事第二日败者组 iG.V赛后采访
2015/12/26 DOTA
Django接受前端数据的几种方法总结
2016/11/04 Python
解决pyqt中ui编译成窗体.py中文乱码的问题
2016/12/23 Python
python编程实现随机生成多个椭圆实例代码
2018/01/03 Python
解决pycharm界面不能显示中文的问题
2018/05/23 Python
python3解析库pyquery的深入讲解
2018/06/26 Python
Python爬虫实现简单的爬取有道翻译功能示例
2018/07/13 Python
python求解汉诺塔游戏
2020/07/09 Python
Python如何读取、写入JSON数据
2020/07/28 Python
CSS3绘制不规则图形的一些方法示例
2015/11/07 HTML / CSS
HTML5中外部浏览器唤起微信分享
2020/01/02 HTML / CSS
Pottery Barn阿联酋:购买家具、家居装饰及更多
2019/12/08 全球购物
解释一下抽象方法和抽象类
2016/08/27 面试题
为什么要做架构设计
2015/07/08 面试题
医护人员英文求职信范文
2013/11/26 职场文书
有趣的广告词
2014/03/18 职场文书
补充协议书
2015/01/28 职场文书
主持人开幕词
2015/01/29 职场文书
期中考试后的感想
2015/08/07 职场文书
Jedis操作Redis实现模拟验证码发送功能
2021/09/25 Redis
【DOTA2】高能暴走TK秀!PSG LGD vs ASTER - DPC 2022 WINTER TOUR CN
2022/04/02 DOTA