PHP にインタビューすると、さまざまなタイプの null 値が真かどうかという質問によく遭遇します:
<code>$a = <span>''</span>; $a = null $a = false; <span>if</span>($a){<span>...</span>} <span>if</span>(isset($a)){<span>...</span>} <span>if</span>(empty($a)){<span>...</span>} <span>...</span></code>
次の例から、zend エンジンで if がどのように処理されるかを簡単に見てみましょう:
<code><span><span><?php </span><span>$a</span> = <span>''</span>; <span>//array();</span><span>if</span>(<span>$a</span>){ <span>echo</span><span>"Y"</span>; }</span></span></code>
これは例です。比較的単純ですが、結果は何もありません。 (この記事に含まれるコードはすべて php-7.0.4 のバージョンです)
前回の記事では、zend 実行フェーズの入り口である zend_execute 関数を紹介しました。慣れていない場合は、ここから直接始めます。前回の記事が読めます。
コンパイルによって生成されるオペコードは次のとおりです:
このうち、opcode=38 が $a = ” の実行演算、opcode=43 が if の演算です
を見てみましょう。
オペコードと 2 つのオペランド タイプに従って、対応するハンドラーが見つかります。 ZEND_JMPZ_SPEC_CV_HANDLER
<code><span>//zend_vm_execute.h #28307</span><span>static</span> ZEND_OPCODE_HANDLER_RET ZEND_FASTCALL ZEND_JMPZ_SPEC_CV_HANDLER(ZEND_OPCODE_HANDLER_ARGS) { USE_OPLINE zval *<span>val</span>; <span>val</span> = _get_zval_ptr_cv_undef(execute_data, opline->op1.var); <span>if</span> (Z_TYPE_INFO_P(<span>val</span>) == IS_TRUE) { ZEND_VM_SET_NEXT_OPCODE(opline + <span>1</span>); ZEND_VM_CONTINUE(); } <span>else</span><span>if</span> (EXPECTED(Z_TYPE_INFO_P(<span>val</span>) if (IS_CV == IS_CV && UNEXPECTED(Z_TYPE_INFO_P(<span>val</span>) == IS_UNDEF)) { SAVE_OPLINE(); GET_OP1_UNDEF_CV(<span>val</span>, BP_VAR_R); ZEND_VM_JMP(OP_JMP_ADDR(opline, opline->op2)); } <span>else</span> { ZEND_VM_SET_OPCODE(OP_JMP_ADDR(opline, opline->op2)); ZEND_VM_CONTINUE(); } } SAVE_OPLINE(); <span>if</span> (i_zend_is_true(<span>val</span>)) { opline++; } <span>else</span> { opline = OP_JMP_ADDR(opline, opline->op2); } <span>if</span> (UNEXPECTED(EG(<span>exception</span>) != NULL)) { HANDLE_EXCEPTION(); } ZEND_VM_JMP(opline); }</code>
この関数から、if の実行プロセスがわかります。条件が true の場合、opline++ は次のオペコードを実行します (つまり、それ以外の場合は、ジャンプが実行され、内側の if ステートメントがスキップされ、外側の if ステートメントが直接実行されます。
i_zend_is_true この関数は、さまざまなタイプの値が true であるかどうかを判断するために使用されます。そうであれば、それは直接処理されます。
<code><span>//</span>zend_operators.h <span>#283</span> static zend_always_inline int i_zend_is_true(zval *op) { int result = <span>0</span>; <span>again</span>: <span>switch</span> (Z_TYPE_P(op)) { <span>case</span><span>IS_TRUE</span>: result = <span>1</span>; <span>break</span>; <span>//</span>数值类型long、double直接判断即可,与c用法相同 <span>case</span><span>IS_LONG</span>: <span>if</span> (Z_LVAL_P(op)) { result = <span>1</span>; } <span>break</span>; <span>case</span><span>IS_DOUBLE</span>: <span>if</span> (Z_DVAL_P(op)) { result = <span>1</span>; } <span>break</span>; <span>//</span>字符串类型根据长度判断:长度><span>1</span>,或=<span>1</span>且不为<span>'0'</span>为<span>true</span>,所以上面那个例子<span>''</span><span> =></span><span>false</span><span>case</span><span>IS_STRING</span>: <span>if</span> (Z_STRLEN_P(op) > <span>1</span> || (Z_STRLEN_P(op) && Z_STRVAL_P(op)[<span>0</span>] != <span>'0'</span>)) { result = <span>1</span>; } <span>break</span>; <span>//</span>数组类型根据数组元素的个数判断:大于<span>0</span>即为真 <span>case</span><span>IS_ARRAY</span>: <span>if</span><span><span>(zend_hash_num_elements(Z_ARRVAL_P(op)))</span> { // <span>(Z_ARRVAL_P(op))</span>-></span>nNumOfElements result = <span>1</span>; } <span>break</span>; <span>case</span><span>IS_OBJECT</span>: result = zend_object_is_true(op); <span>break</span>; <span>//</span>资源类型实际就是整形(后续会专门介绍资源类型),所以直接判断即可 <span>case</span><span>IS_RESOURCE</span>: <span>if</span> (EXPECTED(Z_RES_HANDLE_P(op))) { result = <span>1</span>; } <span>break</span>; <span>//</span>引用类型则根据指向的值判断 <span>case</span><span>IS_REFERENCE</span>: op = Z_REFVAL_P(op); goto again; <span>break</span>; <span>default</span>: <span>break</span>; } <span>return</span> result; }</code>
isset と空の関数は後で追加されます...
').addClass('pre-numbering').hide(); $(this).addClass('has-numbering').parent().append($numbering); for (i = 1; i ').text(i)); }; $numbering.fadeIn(1700); }); });以上、文法:IF判定の実装についての内容を含めて紹介しましたが、PHPチュートリアルに興味のある方の参考になれば幸いです。