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 相关文章推荐
一个简单的自动发送邮件系统(一)
Oct 09 PHP
php IP及IP段进行访问限制的代码
Dec 17 PHP
php+mysql 实现身份验证代码
Mar 24 PHP
在WAMP环境下搭建ZendDebugger php调试工具的方法
Jul 18 PHP
PHP 日,周,月点击排行统计
Jan 11 PHP
php防注入及开发安全详细解析
Aug 09 PHP
php 检查电子邮件函数(自写)
Jan 16 PHP
PHP关于htmlspecialchars、strip_tags、addslashes的解释
Jul 04 PHP
php数组去除空值函数分享
Feb 02 PHP
PHP 5.3和PHP 5.4出现FastCGI Error解决方法
Feb 12 PHP
cakephp打印sql语句的方法
Feb 13 PHP
php操作redis数据库常见方法实例总结
Feb 20 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
PHP 长文章分页函数 带使用方法,不会分割段落,翻页在底部
2009/10/22 PHP
PHP 防注入函数(格式化数据)
2011/08/08 PHP
php微信公众号开发之快递查询
2018/10/20 PHP
用脚本调用样式的几种方法
2006/12/09 Javascript
jQuery编写widget的一些技巧分享
2010/10/28 Javascript
autoPlay 基于jquery的图片自动播放效果
2011/12/07 Javascript
Prototype源码浅析 Enumerable部分(二)
2012/01/18 Javascript
js通过location.search来获取页面传来的参数
2014/09/11 Javascript
node.js集成百度UE编辑器
2015/02/05 Javascript
JavaScript控制浏览器全屏及各种浏览器全屏模式的方法、属性和事件
2015/12/20 Javascript
用JavaScript获取页面文档内容的实现代码
2016/06/10 Javascript
jQuery插件扩展实例【添加回调函数】
2016/11/26 Javascript
详解vue2.0组件通信各种情况总结与实例分析
2017/03/22 Javascript
vue 封装自定义组件之tabal列表编辑单元格组件实例代码
2017/09/07 Javascript
webpack配置导致字体图标无法显示的解决方法
2018/03/06 Javascript
详解微信小程序实现仿微信聊天界面(各种细节处理)
2019/02/17 Javascript
jsonp格式前端发送和后台接受写法的代码详解
2019/11/07 Javascript
详解如何在Javascript和Sass之间共享变量
2019/11/13 Javascript
前端开发之便利店收银系统代码
2019/12/27 Javascript
解决 window.onload 被覆盖的问题方法
2020/01/14 Javascript
node.js文件的复制、创建文件夹等相关操作
2021/02/05 Javascript
[05:59]带你看看DPC的台前幕后
2021/03/11 DOTA
python进程管理工具supervisor使用实例
2014/09/17 Python
Python中将字典转换为XML以及相关的命名空间解析
2015/10/15 Python
基于pytorch padding=SAME的解决方式
2020/02/18 Python
pycharm2020.2 配置使用的方法详解
2020/09/16 Python
Python SMTP发送电子邮件的示例
2020/09/23 Python
Pyecharts 中Geo函数常用参数的用法说明
2021/02/01 Python
int和Integer有什么区别
2013/05/25 面试题
汽车工程专业应届生求职信
2013/10/19 职场文书
宣传工作经验材料
2014/06/02 职场文书
餐厅周年庆活动方案
2014/08/25 职场文书
小学班主任自我评价
2015/03/11 职场文书
教导处教学工作总结
2015/08/12 职场文书
大学生各类奖学金申请书
2019/06/24 职场文书
Java面试题冲刺第十六天--消息队列
2021/08/07 面试题