编程语言
首页 > 编程语言> > PHP:为什么零长度字符串上的数组语法将字符串转换为数组?

PHP:为什么零长度字符串上的数组语法将字符串转换为数组?

作者:互联网

PHP中,您可以使用数组语法来访问字符串索引.以下程序

<?php
$foo = "Hello";
echo $foo[0],"\n";
?>

回声

H

但是,如果您访问零长度字符串的第一个字符

<?php
$bar = "";
$bar[0] = "test";
var_dump($bar);
?>

PHP将您的字符串转换为数组.上面的代码产生

array(1) {
    [0] =>
    string(4) "test"
}

即我的零长度字符串被强制转换为数组.类似的“访问字符串的未定义索引”示例不会产生此转换行为.

$bar = " ";
$bar[1] = "test";
var_dump($bar);

产生字符串t.即$bar仍然是一个字符串,并且不会转换为数组.

当语言需要推断和/或自动为你投射变量时,我会发现这些不直观的边缘情况是不可避免的,但有谁知道幕后发生了什么?

即,在PHP的C/C++级别发生了什么来实现这一目标.为什么我的变量变成了一个数组.

PHP 5.6,如果重要的话.

解决方法:

在C级别,当使用[]运算符完成赋值时,变量将转换为数组.当然,当它是一个字符串时,长度为0并且不是未设置的调用类型(例如,unset($test [0])).

case IS_STRING: {
                zval tmp;

                if (type != BP_VAR_UNSET && Z_STRLEN_P(container)==0) {
                    goto convert_to_array;
                }

https://github.com/php/php-src/blob/PHP-5.6.0/Zend/zend_execute.c#L1156

布尔值假值发生相同的转换.

case IS_BOOL:
            if (type != BP_VAR_UNSET && Z_LVAL_P(container)==0) {
                goto convert_to_array;
            }

通过使用测试确认:

<?php
$bar = false;
$bar[0] = "test";
var_dump($bar);

输出:

array(1) { [0]=> string(4) "test" }

使用true时:

<?php
$bar = true;
$bar[0] = "test";
var_dump($bar);

输出:

WARNING Cannot use a scalar value as an array on line number 3
bool(true)

https://github.com/php/php-src/blob/PHP-5.6.0/Zend/zend_execute.c#L1249

当值是bool类型且值为true时,将执行以下代码:

case IS_BOOL:
            if (type != BP_VAR_UNSET && Z_LVAL_P(container)==0) {
                goto convert_to_array;
            }
            /* break missing intentionally */

        default:
            if (type == BP_VAR_UNSET) {
                zend_error(E_WARNING, "Cannot unset offset in a non-array variable");
                result->var.ptr_ptr = &EG(uninitialized_zval_ptr);
                PZVAL_LOCK(EG(uninitialized_zval_ptr));
            } else { // Gets here when boolean value equals true.
                zend_error(E_WARNING, "Cannot use a scalar value as an array");
                result->var.ptr_ptr = &EG(error_zval_ptr);
                PZVAL_LOCK(EG(error_zval_ptr));
            }
            break;

PHP 5.6版使用ZEND版本2.6.0

标签:php,types,casting,php-internals
来源: https://codeday.me/bug/20190712/1443441.html