<p>今天在公司实现一个模块功能时写了以下代码:</p>php
class ProductCategory { const TYPES = [ 1 => 'type1', 2 => 'type2', ]; public function getType() { return isset(self::TYPES[$this->type]) ? self:TYPES[$this->type] : 'unrecognized_type'; } }
<p>竟然报错, 在编译阶段就通不过了.</p> <blockquote>Fatal error: Cannot use isset() on the result of an expression (you can use "nul<br>l !== expression" instead)</blockquote> <p>错误信息意思很明显, 但个人代码isset里面并非一个表达式啊,这让我百思不得其解.<br>我带着疑惑在家里从新敲下了如上代码,编译经过, 正常运行. <br><code>php -v</code>查看版本, 7.1. 而公司的开发机上运行的是php5.6</p> <p>那么,为何会形成这样的差别呢?只能翻看源码看isset的底层实现.</p> <p>众所周知, isset不是函数, 而是语法结构, 那么若是发生错误, 在编译阶段就会出错.</p> <p>对比一下php5.6和php7.0+版本的<code>zend_language_parse.y</code></p> <p>在php5.6版本中的<code>zend_language_parse.y</code>的1283行</p> ``` isset_variable: variable { zend_do_isset_or_isempty(ZEND_ISSET, &$$, &$1 TSRMLS_CC); } | expr_without_variable { zend_error_noreturn(E_COMPILE_ERROR, "Cannot use isset() on the result of an expression (you can use \"null !== expression\" instead)"); } ; ```express
<p>很明显,在词法解析的时候, 类常量被定义成非变量了</p> <p>看一看<code>expr_without_variable</code>的定义, 在该文件的776行到858行, 咱们找到了这样一个定义:</p> ``` | combined_scalar_offset { zend_do_end_variable_parse(&$1, BP_VAR_R, 0 TSRMLS_CC); } ```segmentfault
<p>再看<code>combined_scalar_offset </code>的定义:</p> ``` general_constant '[' dim_offset ']' { zend_do_begin_variable_parse(TSRMLS_C); fetch_array_dim(&$$, &$1, &$3 TSRMLS_CC); } ```php7
<p>再看<code>general_constant </code>的定义:</p> ``` class_constant { $$ = $1; } ```函数
<p>恍然大悟, 类常量被定义为非变量, 因此抛出编译错误. </p> <p>而在php7.0+版本</p> ``` combined_scalar_offset { zend_do_end_variable_parse(&$1, BP_VAR_R, 0 TSRMLS_CC); } ```fetch
<p>是被去掉了的. 因此编译经过, 并成功运行.</p> <p>也不知道这个算是bug, 仍是5.6的<code>feature</code>~~~</p>this
原文地址:http://www.javashuo.com/article/p-kugrhfbh-bz.htmlscala