isset在php5.6-和php7.0+的一些差別

<p>今天在公司實現一個模塊功能時寫了以下代碼:</p>php

class ProductCategory
{
    const TYPES = [
        1 =&gt; 'type1',
        2 =&gt; 'type2',  
    ];
    
    public function getType()
    {
        return isset(self::TYPES[$this-&gt;type]) ? self:TYPES[$this-&gt;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, &amp;$$, &amp;$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(&amp;$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(&amp;$$, &amp;$1, &amp;$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(&amp;$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

相關文章
相關標籤/搜索