PHP数组Key强制类型转换实现原理解析
作者:硬核项目经理
PHP是弱类型语言,就像JavaScript一样,在定义变量时,不需要强制指定变量的类型。同时,PHP又有着强大的数组功能,数组的Key即可以是普通的数字类型下标,也可以是字符串类型的Hash键值,那么,当一个数组的Key同时拥有字符串和数字时,会产生什么情况呢?
首先来看下面这样一段代码:
$arr = [ "1" => "a", "01" => "b", 1 => "aa", 1.1 => "aaa", "0.1" => "bb", ]; var_dump($arr); // array(3) { // [1] => // string(3) "aaa" // '01' => // string(1) "b" // '0.1' => // string(2) "bb" // }
咦?我们定义的"1"、1下标的值都变成了1.1的"aaa"了?
没错,PHP中的数组Key值只接受数字和字符串类型,当Key是字符串时,会强制转换为数字类型,遵守类型强制转换的规则。浮点数也是同样的道理,直接转换成了向下取整的整型。
那么"0.1"和"01"为什么还在?首先,"01"不是标准的十进制数值,无法转换成整型,所以"01"还是一个字符串下标,那"0.1"呢?它当然也不是一个标准的十进制数值。这里是违背了字符串转型数字的强制类型转换原则的,在变量的强制转换中,这两种字符串都会被转换为0,但在数组中则不会,这里会是一个坑,也是需要注意的地方。
在PHP官方文档中给出的Key值转换说明如下:
- 包含有合法整型值的字符串会被转换为整型。例如键名 "8" 实际会被储存为 8。但是 "08" 则不会强制转换,因为其不是一个合法的十进制数值。
- 浮点数也会被转换为整型,意味着其小数部分会被舍去。例如键名 8.7 实际会被储存为 8。
- 布尔值也会被转换成整型。即键名 true 实际会被储存为 1 而键名 false 会被储存为 0。
- Null 会被转换为空字符串,即键名 null 实际会被储存为 ""。
- 数组和对象不能被用为键名。坚持这么做会导致警告:Illegal offset type。
接下来,是笔者曾经做过的一道面试题,和这个类型转换有着非常大的关系,代码如下:
a = ['a']; $a[2] = 'b'; $a[] = 'c'; $a['1'] = 'd'; // 以下循环的输出结果是? foreach ($a as $v) { echo $v, ','; } // 以下循环的输出结果是? for ($i = 0; $i < count($a); ++$i) { echo $a[$i], ' ,'; }
大家先不要运行,直接看代码看看能不能看出这两段代码的输出结果会是什么,然后运行一下,看看结果和你想像的是不是一样。
测试代码: https://github.com/zhangyue0503/dev-blog/blob/master/php/201910/source/关于PHP数组Key的强制转换.php
参考资料: https://www.php.net/manual/zh/language.types.array.php
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。