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


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 相关文章推荐
用ADODB来让PHP操作ACCESS数据库的方法
Dec 31 PHP
一个简单php扩展介绍与开发教程
Aug 19 PHP
10 个经典PHP函数
Oct 17 PHP
PHP中执行MYSQL事务解决数据写入不完整等情况
Jan 07 PHP
模板引擎smarty工作原理以及使用示例
May 25 PHP
php中使用array_filter()函数过滤空数组的实现代码
Aug 19 PHP
php查询相似度最高的字符串的方法
Mar 12 PHP
ThinkPHP3.2.3实现分页的方法详解
Jun 03 PHP
详解PHP归并排序的实现
Oct 18 PHP
浅谈socket同步和异步、阻塞和非阻塞、I/O模型
Dec 15 PHP
php5.5使用PHPMailer-5.2发送邮件的完整步骤
Oct 14 PHP
php实现断点续传大文件示例代码
Jun 19 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
apache和php之间协同工作的配置经验分享
2013/04/08 PHP
Yii中表单用法实例详解
2016/01/05 PHP
tp5框架的增删改查操作示例
2019/10/31 PHP
php操作redis数据库常见方法实例总结
2020/02/20 PHP
php设计模式之组合模式实例详解【星际争霸游戏案例】
2020/03/27 PHP
Raphael带文本标签可拖动的图形实现代码
2013/02/20 Javascript
jquery显示和隐藏div特效实例
2013/02/27 Javascript
JS获取后台Cookies值的小例子
2013/03/04 Javascript
jQuery 全选/反选以及单击行改变背景色实例
2013/07/02 Javascript
jqeury-easyui-layout问题解决方法
2014/03/24 Javascript
JS实现在网页中弹出一个输入框的方法
2015/03/03 Javascript
js+html5通过canvas指定开始和结束点绘制线条的方法
2015/06/05 Javascript
JavaScript 数组- Array的方法总结(推荐)
2016/07/21 Javascript
JS正则表达式之非捕获分组用法实例分析
2016/12/28 Javascript
JQuery统计input和textarea文字输入数量(代码分享)
2016/12/29 Javascript
mui上拉加载更多下拉刷新数据的封装过程
2017/11/03 Javascript
浅谈React组件之性能优化
2018/03/02 Javascript
layui中使用jquery控制radio选中事件的示例代码
2018/08/15 jQuery
vue eslint简要配置教程详解
2019/07/26 Javascript
vue-router 2.0 跳转之router.push()用法说明
2020/08/12 Javascript
小议Python中自定义函数的可变参数的使用及注意点
2016/06/21 Python
Python3 伪装浏览器的方法示例
2017/11/23 Python
Python3.5基础之函数的定义与使用实例详解【参数、作用域、递归、重载等】
2019/04/26 Python
Python图像处理模块ndimage用法实例分析
2019/09/05 Python
Python数据库小程序源代码
2019/09/15 Python
使用Pandas将inf, nan转化成特定的值
2019/12/19 Python
Python使用Tkinter实现滚动抽奖器效果
2020/01/06 Python
Python3如何实现Win10桌面自动切换
2020/08/11 Python
大学活动策划书范文
2014/01/10 职场文书
教师个人剖析材料
2014/02/05 职场文书
房地产开盘策划方案
2014/02/10 职场文书
党员四风问题对照检查材料
2014/09/27 职场文书
万能检讨书开头与结尾怎么写
2015/02/17 职场文书
python 如何执行控制台命令与操作剪切板
2021/05/20 Python
十大最帅动漫男主 碓冰拓海上榜,第一是《灌篮高手》男主角
2022/03/18 日漫
GO语言异常处理分析 err接口及defer延迟
2022/04/14 Golang