PHP 中模糊的函数调用行为:揭开隐藏的细微差别
在 PHP 中,调用函数并封装其结果时会出现一种奇怪的行为括号内。这种看似无害的操作可能会改变结果的解释,从而可能导致意外行为。
考虑以下代码片段:
error_reporting(E_ALL | E_STRICT); function get_array() { return array(); } function foo() { return reset(get_array()); // Error: "Only variables should be passed by reference" return reset((get_array())); // Success } foo();
在第一个变体中,尝试分配以下结果: reset() 函数的 get_array() 会触发错误,因为它应该是通过引用传递的变量。然而,当我们将函数调用括在括号中时,脚本会成功运行。
虽然人们可能会推测底层机制,但对文档的彻底检查无法为这种现象提供明确的解释。
深入研究 PHP 语法的复杂性并使用 phc 等工具可视化代码的 AST,发现两个片段都会生成相同的解析树。这消除了语法更改影响行为的可能性。
但是,仔细检查 PHP 操作码结构会发现 ZEND_SEND_VAR_NO_REF 操作码中隐藏的警告。此操作码规定,当参数不是函数调用且引用计数为 1 时,不应引发引用错误。
在 foo() 的第二个变体中,由于不再识别函数调用到添加的括号。因此,系统将参数分类为 expr_without_variable 并启动 SEND_VAR_NO_REF 操作码。由于返回的数组的引用计数为 1,因此它被认为适合作为变量,从而避免引用错误。
需要注意的是,此行为应被视为错误而不是预期功能。依赖这种不一致可能会导致不可预测且不可靠的代码。
以上是为什么在 PHP 中将函数调用括在括号中可以避免引用错误?的详细内容。更多信息请关注PHP中文网其他相关文章!