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

   2015-11-07 0
核心提示:以下是对数组非数字键名引号的必要性进行了详细的分析介绍,需要的朋友可以过来参考下
我看到过很多人操作数组的时候, 对于数组中的非数字键名不使用引号
复制代码 代码如下:

  $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);//分配空间,生成字符串
}
....
 
标签: 数组 键名 引号
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • Swift数组的加法运算符用法:array1 += array2
    var stringList1 = [String]()//创建String类型空数组var stringList2 = ["1", "3", "5", "7", "zoo", "9","zoo"]var stringList3 :[String] = ["2", "4", "6","apple&q
    02-09
  • ">C# 使用SIMD向量类型加速浮点数组求和运算(1
    作者: 目录一、缘由二、使用向量类型2.1 基本算法2.2 使用大小固定的向量(如 Vector4)2.2.1 介绍2.2.2 用Vector4编写浮点数组求和函数2.3 使用大小与硬件相关的向量(如 VectorT)2.3.1 介绍2.2.1.1 使用经验2.3.2 用 VectorT 编写浮点数组求和函数三、搭
    02-09
  • 一个把String转成byte数组的小程序 java string
    来源:http://www.study-code.com/java/j2se/66420.htm /**@类功能:将一个文件中的String声明转换成byte[] 声明***@作者:关文柏*@时间:2006年11月11日*///---------------------------------------------------------------////NOTE: Make sure the identifi
    02-09
  • 【Rust】字节数组 rust 字符串
    环境Rust 1.56.1VSCode 1.61.2概念参考:https://doc.rust-lang.org/rust-by-example/std/str.html示例在字符串前加上一个 b 来表示。main.rsuse std::str;fn main() {let bytestring = b"this is a byte string";println!("A byte string: {:?}", bytestring
    02-09
  • 【小实验】rust的数组是在堆上分配还是在栈上分
    先看代码: fn main(){ let v = [1,2,3,4,5]; let addr = v[0] as *const i32 as usize;println!("arr={}, addr=0x{:X}", v.len(), addr); // let top = 1; let addr1 = top as *const i32 as usize; println!("stack top:0x{:X}", addr1);}编译:rustc array
    02-09
  • 【Rust】可变数组 rust二维数组
    环境Rust 1.56.1VSCode 1.61.2概念参考:https://doc.rust-lang.org/rust-by-example/std/vec.html示例可变数组(Vector)存储在堆上和普通数组的区别是长度可变。main.rsfn main() {let collected_iterator: Veci32 = (0..10).collect();println!("Collected
    02-09
  • Rust 旋转数组
    经典三旋转:旋转数组经典算法就是三旋转先整体旋转之后在局部旋转需要注意 求余运算,超过数组长度后要取余数后在旋转如:[1,2,3,4,5,6,7]  3     [7,6,5,4,3,2,1]  整体旋转     [5,6,7,4,3,2,1]  [..k] 旋转     [5,6,7,1,2,3,4]  [k..]
    02-09
  • Delphi 的内存操作函数1-2: 给数组指针分配内存
    GetMemAllocMemReallocMemFreeMemGetMemoryReallocMemoryFreeMemoryNewDisposeNewStrDisposeStrStrNewStrAllocStrDisposeGlobalAllocPtrGlobalFreePtrWideStrAllocAnsiStrAllocStrDisposeMoveMoveMemoryCopyMemoryZeroMemoryFillMemoryFillCharStrBufSize给字
    02-09
  • Delphi 变体数组 Dataset Locate 查找定位
    Format 函数 Delphi 支持“开参数”和动态数组,变体数组,使用时的语法类似 Delphi 中的集合:采用两个方括号把不同类型的变量括起来(这太方便了啊),也可以采用声明一个 TVarRec 类型的数组来容纳不同的类型变量(具体请参阅其帮助文档)Format函数声明比
    02-09
  • Delphi指针总结 delphi 数组指针
    看一个指针用法的例子:      1          var      2              X,    Y:    Integer;        //    X    and    Y    整数类型      3              P:    ^Integer;  
    02-09
点击排行