解析数组非数字键名引号的必要性


Posted in PHP onAugust 09, 2013

我看到过很多人操作数组的时候, 对于数组中的非数字键名不使用引号

  $array[key] = $value;

我可以理解有些人可能会觉得这样的代码很”整洁”, 并且也能正常执行.
更甚至,如果他很”幸运的”php配置的好:
error_reporting = ~E_NOTIC

他也许永远都沉浸在自己的”整洁”风格中, 看不到任何的NOTICE提示, 也不会意识到, 他这么做, 能损失多少的性能~
来, 我们一起来看看:
good.php:
<?php
   $array = array();
   $i = 0;
   while(++$i < 1000){
       $array['good'] = 2;
   }
?>

bad.php:
<?php
   $array = array();
   $i = 0;
   while(++$i < 1000){
       $array[good] = 2;
   }
?>

分别看运行时间(多次平均时间):
加引号的:
$ time php -f good.php
real 0m0.013s
user 0m0.005s
sys 0m0.007

不加引号的:
$ time php -f bad.php
PHP Notice: Use of undefined constant bad - assumed 'bad' in /home/huixinchen/tmp/bad.php
on line (此处省略999行NOTICE)
real 0m0.100s
user 0m0.020s
sys 0m0.029

看看,差别有多大?
哦, 或许我们应该模拟一下那些”幸运的”人们的情况, 去掉花费在记录NOTICE的开销, 看看~
$ time php -f bad.php
real 0m0.037s
user 0m0.018s
sys 0m0.018

我们可以看出, 基本上, 使用引号,和不使用引号的效率损失在3倍以上
那么, 这些效率损失到哪里去了呢?
我们分别看下, 俩个文件生成的OPCODE序列:
good.php :
filename: /home/huixinchen/tmp/good.php
compiled vars: !0 = $array, !1 = $i
line # op fetch ext return operands
-------------------------------------------------------------------------------
   2 0 INIT_ARRAY ~0
         1 ASSIGN !0, ~0
   3 2 ASSIGN !1, 0
   4 3 PRE_INC $3 !1
         4 IS_SMALLER ~4 $3, 1000
         5 JMPZ ~4, ->9
   5 6 ZEND_ASSIGN_DIM !0, 'good'
         7 ZEND_OP_DATA 2, $6
   6 8 JMP ->3
   8 9 RETURN 1
        10* ZEND_HANDLE_EXCEPTIO

bad.php :
filename: /home/huixinchen/tmp/bad.php
compiled vars: !0 = $array, !1 = $i
line # op fetch ext return operands
-------------------------------------------------------------------------------
   2 0 INIT_ARRAY ~0
         1 ASSIGN !0, ~0
   3 2 ASSIGN !1, 0
   4 3 PRE_INC $3 !1
         4 IS_SMALLER ~4 $3, 1000
         5 JMPZ ~4, ->10
   5 6 FETCH_CONSTANT ~5 'bad'
         7 ZEND_ASSIGN_DIM !0, ~5
         8 ZEND_OP_DATA 2, $7
   6 9 JMP ->3
   8 10 RETURN 1
        11* ZEND_HANDLE_EXCEPTIO

我们可以看出(其实,根据NOTICE的提示也知道), PHP会把没有引号引起来的键名当作是常量去获取, 当找不到的时候, 抛出一个NOTICE, 然后再根据”常量明”生成一个字符串, 然后再讲这个字符串做为键名继续~
聪明的你一定会想到, 可能会出现如下不可预期的错误:
define('key_name' , 'laruence');
....
//省略很多行代码
$array[key_name] = 2; //变成了 $array['laruence'] = 2;
//这样的错误, 你会很郁闷吧?

明白了么? 数组中的非数字键的键名一定要有引号啊~
哦, 还记得有人会说, 那在字符串变量替换的时候, 写引号会导致错误,
恩, 标准写法:
$string = "variable value is {$array['key']}"

我很赞同:”be lazy”, 但是, lazy也是应该有原则的.
最后, 好的代码,不应该通过关闭error_reporting来伪装.
附注, FETCH_CONSTANT OPCODE中找不到常量的相关逻辑:
....
if (!zend_get_constant(opline->op2.u.constant.value.str.val,
     opline->op2.u.constant.value.str.len, &EX_T(opline->result.u.var).tmp_var TSRMLS_CC)) {
       zend_error(E_NOTICE, "Use of undefined constant %s - assumed '%s'",
                opline->op2.u.constant.value.str.val,
                opline->op2.u.constant.value.str.val);
       EX_T(opline->result.u.var).tmp_var = opline->op2.u.constant;//获取"常量"名字符串
       zval_copy_ctor(&EX_T(opline->result.u.var).tmp_var);//分配空间,生成字符串
}
....
PHP 相关文章推荐
PHP目录函数实现创建、读取目录教程实例
Jan 13 PHP
自己写了一个php检测文件编码的函数
Apr 21 PHP
PHPer 需要了解的 5 个 Composer 小技巧
Aug 18 PHP
php curl登陆qq后获取用户信息时证书错误
Feb 03 PHP
浅谈php命令行用法
Feb 04 PHP
thinkPHP学习笔记之安装配置篇
Mar 05 PHP
又十个超级有用的PHP代码片段
Sep 24 PHP
[原创]ThinkPHP让../Public在模板不解析(直接输出)的方法
Oct 09 PHP
浅谈PHP拦截器之__set()与__get()的理解与使用方法
Oct 18 PHP
php中static和const关键字用法分析
Dec 07 PHP
PHP进程通信基础之信号量与共享内存通信
Feb 19 PHP
PHP插件PHPMailer发送邮件功能
Feb 28 PHP
php防注入及开发安全详细解析
Aug 09 #PHP
分割GBK中文遭遇乱码的解决方法
Aug 09 #PHP
解析isset与is_null的区别
Aug 09 #PHP
PHP中怎样保持SESSION不过期 原理及方案介绍
Aug 08 #PHP
php中用socket模拟http中post或者get提交数据的示例代码
Aug 08 #PHP
浅析php变量作用域的一些问题
Aug 08 #PHP
解析php开发中的中文编码问题
Aug 08 #PHP
You might like
外媒评选出10支2020年最受欢迎的Dota2战队
2021/03/05 DOTA
PHP 解决session死锁的方法
2013/06/20 PHP
php使用百度翻译api示例分享
2014/01/31 PHP
浅谈php7的重大新特性
2015/10/23 PHP
PHP实现带重试功能的curl连接示例
2016/07/28 PHP
thinkphp 手机号和用户名同时登录
2017/01/20 PHP
JavaScript实际应用:innerHTMl和确认提示的使用
2006/06/22 Javascript
DOMAssitant最新版 DOMAssistant 2.5发布
2007/12/25 Javascript
用js实现层随着内容大小动态渐变改变 推荐
2009/12/19 Javascript
JS Replace()的高级使用方法介绍
2013/06/29 Javascript
利用jq让你的div居中的好方法分享
2013/11/21 Javascript
javaScript中两个等于号和三个等于号之间的区别介绍
2014/06/27 Javascript
javascript实现树形菜单的方法
2015/07/17 Javascript
jquery事件的ready()方法使用详解
2015/11/11 Javascript
纯JavaScript代码实现文本比较工具
2016/02/17 Javascript
浏览器兼容的JS写法总结
2016/04/27 Javascript
ionic实现带字的toggle滑动组件
2016/08/27 Javascript
jQuery UI制作选项卡(tabs)
2016/12/13 Javascript
promise处理多个相互依赖的异步请求(实例讲解)
2017/08/03 Javascript
Angular使用动态加载组件方法实现Dialog的示例
2018/05/11 Javascript
Javascript获取某个月的天数
2018/05/30 Javascript
Vue中 v-if/v-show/插值表达式导致闪现的原因及解决办法
2018/10/12 Javascript
Angular6新特性之Angular Material
2018/12/28 Javascript
零基础之Node.js搭建API服务器的详解
2019/03/08 Javascript
Python下线程之间的共享和释放示例
2015/05/04 Python
图文详解python安装Scrapy框架步骤
2019/05/20 Python
react+django清除浏览器缓存的几种方法小结
2019/07/17 Python
PYTHON EVAL的用法及注意事项解析
2019/09/06 Python
Pytorch基本变量类型FloatTensor与Variable用法
2020/01/08 Python
Python脚本破解压缩文件口令实例教程(zipfile)
2020/06/14 Python
财务会计专业毕业生自荐信
2013/10/19 职场文书
电子商务专业学生的学习自我评价
2013/10/27 职场文书
《飞向蓝天的恐龙》教学反思
2014/04/09 职场文书
高中生评语大全
2014/04/25 职场文书
工伤事故赔偿协议书(标准)
2014/09/29 职场文书
新生入学欢迎词
2015/01/26 职场文书