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


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 相关文章推荐
动易数据转成dedecms的php程序
Apr 07 PHP
为PHP初学者的8点有效建议
Nov 20 PHP
php的chr和ord函数实现字符加减乘除运算实现代码
Dec 05 PHP
PHP版网站缓存加快打开速度的方法分享
Jun 03 PHP
PHP采用XML-RPC构造Web Service实例教程
Jul 16 PHP
如何批量清理系统临时文件(语言:C#、 C/C++、 php 、python 、java )
Feb 01 PHP
Zend Framework教程之Zend_Registry对象用法分析
Mar 22 PHP
CI框架文件上传类及图像处理类用法分析
May 18 PHP
PHP使用DOM对XML解析处理操作示例
Jul 04 PHP
Laravel实现ORM带条件搜索分页
Oct 24 PHP
php模拟实现斗地主发牌
Apr 22 PHP
php中yii框架实例用法
Dec 22 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
77A一级收信机修理记
2021/03/02 无线电
MayFish PHP的MVC架构的开发框架
2009/08/13 PHP
php自定义函数之递归删除文件及目录
2010/08/08 PHP
获取用户Ip地址通用方法与常见安全隐患(HTTP_X_FORWARDED_FOR)
2013/06/01 PHP
PHP session文件独占锁引起阻塞问题解决方法
2015/05/12 PHP
PHP文件下载实例代码浅析
2016/08/17 PHP
PHP获取用户客户端真实IP的解决方案
2016/10/10 PHP
Extjs学习笔记之五 一个小细节renderTo和applyTo的区别
2010/01/07 Javascript
使用jquery.upload.js实现异步上传示例代码
2014/07/29 Javascript
Vue.js中extend选项和delimiters选项的比较
2017/07/17 Javascript
Node.JS 循环递归复制文件夹目录及其子文件夹下的所有文件
2017/09/18 Javascript
使用vue-cli编写vue插件的方法
2018/02/26 Javascript
vue中echarts3.0自适应的方法
2018/02/26 Javascript
Vue.js 中的 v-show 指令及用法详解
2018/11/19 Javascript
微信小程序MUI导航栏透明渐变功能示例(通过改变opacity实现)
2019/01/24 Javascript
Vue-cli3多页面配置详解
2020/03/22 Javascript
Python运行的17个时新手常见错误小结
2012/08/07 Python
python搭建简易服务器分析与实现
2012/12/15 Python
下载给定网页上图片的方法
2014/02/18 Python
Python的gevent框架的入门教程
2015/04/29 Python
python flask 如何修改默认端口号的方法步骤
2019/07/12 Python
python tornado修改log输出方式
2019/11/18 Python
pytorch 常用函数 max ,eq说明
2020/06/28 Python
Django自带的用户验证系统实现
2020/12/18 Python
德国网上药房:Apotal
2017/04/04 全球购物
乡镇庆八一活动方案
2014/02/02 职场文书
仓库管理计划书
2014/05/04 职场文书
自我推荐信范文
2014/05/09 职场文书
材料物理专业求职信
2014/09/01 职场文书
2014年学校领导班子对照检查材料
2014/09/19 职场文书
党的群众路线教育实践活动个人对照检查剖析材料
2014/09/23 职场文书
借款民事起诉状范文
2015/05/19 职场文书
小学班长竞选稿
2015/11/20 职场文书
解决Navicat for Mysql连接报错1251的问题(连接失败)
2021/05/27 MySQL
压缩Redis里的字符串大对象操作
2021/06/23 Redis