목차
[PHP源码阅读]empty和isset函数,emptyisset
函数使用格式
empty
isset
参数说明
运行示例
找到函数的定义位置
函数执行步骤

源码解读
小结
백엔드 개발 PHP 튜토리얼 [PHP源码阅读]empty和isset函数,emptyisset_PHP教程

[PHP源码阅读]empty和isset函数,emptyisset_PHP教程

Jul 12, 2016 am 08:51 AM
php

[PHP源码阅读]empty和isset函数,emptyisset

近日被问到PHP中empty和isset函数时怎么判断变量的,刚开始我是一脸懵逼的,因为我自己也只是一知半解,为了弄懂其真正的原理,赶紧翻开源码研究研究。经过分析可发现两个函数调用的都是同一个函数,因此本文将对两个函数一起分析。

我在github有对PHP源码更详细的注解。感兴趣的可以围观一下,给个star。PHP5.4源码注解。可以通过commit记录查看已添加的注解。

函数使用格式

empty

<p>bool empty ( mixed $var )</p>
로그인 후 복사


判断变量是否为空。

isset

<p>bool isset ( mixed $var [ , mixed $... ] )</p>


로그인 후 복사

判断变量是否被设置且不为NULL。

参数说明

对于empty,在PHP5.5版本以前,empty只支持变量参数,其他类型的参数会导致解析错误,比如函数调用的结果不能作为参数。

对于isset,如果变量被如unset的函数设为NULL,则函数会返回false。如果多个参数被传递到isset函数,那么只有所有参数都被设置isset函数才会返回true。从左到右计算,一旦遇到没被设置的变量就停止。

运行示例

<span>$result</span> = <span>empty</span>(0); <span>//</span><span> true</span>
<span>$result</span> = <span>empty</span>(<span>null</span>); <span>//</span><span> true<br /></span><span>$result</span> = <span>empty</span>(<span>false</span>); <span>//</span><span> true</span>
<span>$result</span> = <span>empty</span>(<span>array</span>()); <span>//</span><span> true</span>
<span>$result</span> = <span>empty</span>('0'); <span>//</span><span> true</span>
<span>$result</span> = <span>empty</span>(1); <span>//</span><span> false</span>
<span>$result</span> = <span>empty</span>(<span>callback function</span>); <span>//</span><span> 报错<br /><br />$a = null;<br />$result = isset($a); // false;<br /><br />$a = 1;<br />$result = isset($a); // true;<br /><br />$a = 1;$b = 2;$c = 3;<br />$result = isset($a, $b, $c); // true<br /><br /></span>
로그인 후 복사
$a = 1;$b = null;$c = 3;<br />$result = isset($a, $b, $c); // false
로그인 후 복사

找到函数的定义位置

实际上,empty不是一个函数,而是一个语言结构。语言结构是在PHP程序运行前编译好的,因此不能像之前那样简单地搜索"PHP_FUNCTION empty"或"ZEND_FUNCTION empty"查看其源码。要想看empty等语言结构的源码,先要理解PHP代码执行的机制。

PHP执行代码会经过4个步骤,其流程图如下所示:

[PHP源码阅读]empty和isset函数,emptyisset_PHP教程"isset" { return T_ISSET; } "empty" { return T_EMPTY; }


接下来就到了Parsing阶段,这个阶段,程序将T_ISSET和T_EMPTY等Tokens转换成有意义的表达式,此时会做语法分析,Tokens的yacc保存在zend_language_parser.y文件中,可以找到T_ISSET和T_EMPTY的定义:

<span>internal_functions_in_yacc:
T_ISSET </span><span>'</span><span>(</span><span>'</span> isset_variables <span>'</span><span>)</span><span>'</span> { $$ = $<span>3</span><span>; }
</span>| T_EMPTY <span>'</span><span>(</span><span>'</span> variable <span>'</span><span>)</span><span>'</span> { zend_do_isset_or_isempty(ZEND_ISEMPTY, &$$, &$<span>3</span><span> TSRMLS_CC); }
</span>| T_INCLUDE expr { zend_do_include_or_eval(ZEND_INCLUDE, &$$, &$<span>2</span><span> TSRMLS_CC); }
</span>| T_INCLUDE_ONCE expr { zend_do_include_or_eval(ZEND_INCLUDE_ONCE, &$$, &$<span>2</span><span> TSRMLS_CC); }
</span>| T_EVAL <span>'</span><span>(</span><span>'</span> expr <span>'</span><span>)</span><span>'</span> { zend_do_include_or_eval(ZEND_EVAL, &$$, &$<span>3</span><span> TSRMLS_CC); }
</span>| T_REQUIRE expr { zend_do_include_or_eval(ZEND_REQUIRE, &$$, &$<span>2</span><span> TSRMLS_CC); }
</span>| T_REQUIRE_ONCE expr { zend_do_include_or_eval(ZEND_REQUIRE_ONCE, &$$, &$<span>2</span><span> TSRMLS_CC); }
;</span>
로그인 후 복사


isset和empty函数最终都执行了zend_do_isset_or_isempty函数,继续查找
grep -rn "zend_do_isset_or_isempty"
可以发现,此函数在zend_compile.c文件中定义。

函数执行步骤

1、解析参数

2、检查是否为可写变量

3、如果是变量的op_type是IS_CV(编译时期的变量),则设置其opcode为ZEND_ISSET_ISEMPTY_VAR;否则从active_op_array中获取下一个op值,根据其op值设置last_op的opcode。

4、设置了opcode之后,之后会交给zend_excute执行。


源码解读

IS_CV是编译器使用的一种cache机制,这种变量保存着它被引用的变量的地址,当一个变量第一次被引用的时候,就会被CV起来,以后这个变量的引用就不需要再去查找active符号表了。

对于empty函数,到了opcode的步骤后,参阅opcode处理函数,可以知道,isset和empty在excute的时候执行的是ZEND_ISSET_ISEMPTY_VAR等一系列函数,以ZEND_ISSET_ISEMPTY_VAR_SPEC_CV_VAR_HANDLER为例,找到这个函数的定义在zend_vm_execute.h。查看函数可以知道,empty函数的最终执行函数是i_zend_is_true(),而i_zend_is_true函数定义在zend_execute.h。i_zend_is_true函数的核心代码如下:

        <span>switch</span><span> (Z_TYPE_P(op)) {
        </span><span>case</span><span> IS_NULL:
            result </span>= <span>0</span><span>;
            </span><span>break</span><span>;
        </span><span>case</span><span> IS_LONG:
        </span><span>case</span><span> IS_BOOL:
        </span><span>case</span><span> IS_RESOURCE:
            </span><span>//</span><span> empty参数为整数时非0的话就为false</span>
            result = (Z_LVAL_P(op)?<span>1</span>:<span>0</span><span>);
            </span><span>break</span><span>;
        </span><span>case</span><span> IS_DOUBLE:
            result </span>= (Z_DVAL_P(op) ? <span>1</span> : <span>0</span><span>);
            </span><span>break</span><span>;
        </span><span>case</span><span> IS_STRING:
            </span><span>if</span> (Z_STRLEN_P(op) == <span>0</span>
                || (Z_STRLEN_P(op)==<span>1</span> && Z_STRVAL_P(op)[<span>0</span>]==<span>'</span><span>0</span><span>'</span><span>)) {
                </span><span>//</span><span> empty("0") == true</span>
                result = <span>0</span><span>;
            } </span><span>else</span><span> {
                result </span>= <span>1</span><span>;
            }
            </span><span>break</span><span>;
        </span><span>case</span><span> IS_ARRAY:
            </span><span>//</span><span> empty(array) 是根据数组的数量来判断</span>
            result = (zend_hash_num_elements(Z_ARRVAL_P(op))?<span>1</span>:<span>0</span><span>);
            </span><span>break</span><span>;
        </span><span>case</span><span> IS_OBJECT:
            </span><span>if</span>(IS_ZEND_STD_OBJECT(*<span>op)) {
                TSRMLS_FETCH();

                </span><span>if</span> (Z_OBJ_HT_P(op)-><span>cast_object) {
                    zval tmp;
                    </span><span>if</span> (Z_OBJ_HT_P(op)->cast_object(op, &tmp, IS_BOOL TSRMLS_CC) ==<span> SUCCESS) {
                        result </span>=<span> Z_LVAL(tmp);
                        </span><span>break</span><span>;
                    }
                } </span><span>else</span> <span>if</span> (Z_OBJ_HT_P(op)-><span>get</span><span>) {
                    zval </span>*tmp = Z_OBJ_HT_P(op)-><span>get</span><span>(op TSRMLS_CC);
                    </span><span>if</span>(Z_TYPE_P(tmp) !=<span> IS_OBJECT) {
                        </span><span>/*</span><span> for safety - avoid loop </span><span>*/</span><span>
                        convert_to_boolean(tmp);
                        result </span>=<span> Z_LVAL_P(tmp);
                        zval_ptr_dtor(</span>&<span>tmp);
                        </span><span>break</span><span>;
                    }
                }
            }
            result </span>= <span>1</span><span>;
            </span><span>break</span><span>;
        </span><span>default</span><span>:
            result </span>= <span>0</span><span>;
            </span><span>break</span><span>;
    }</span>
로그인 후 복사


这段代码比较直观,函数没有对检测值做任何的转换,通过这段代码来进一步分析示例中的empty函数做分析:
empty(null),到IS_NULL分支,result=0,i_zend_is_true() == 0,!i_zend_is_true() == 1,因此返回true。

empty(false),到IS_BOOL分支,result = ZLVAL_P(false) = 0,i_zend_is_true() == 0,!i_zend_is_true() == 1,因此返回true。

empty(array()),到IS_ARRAY分支,result = zend_hash_num_elements(Z_ARRVAL_P(op)) ? 1 : 0),zend_hash_num_elements返回数组元素的数量,array为空,因此result为0,i_zend_is_true() == 0,!i_zend_is_true() == 1,因此返回true。

empty('0'),到IS_STRING分支,因为Z_STRLENP(op) == 1 且 Z_STRVAL_P(op)[0] == '0',因此result为0,i_zend_is_true() == 0,!i_zend_is_true() == 1,因此返回true。

empty(1),到IS_LONG分支,result = Z_LVAL_P(op) = 1,i_zend_is_true == 1,!i_zend_is_true() == 0,因此返回false。

对于isset函数,最终实现判断的代码是:

if (isset && Z_TYPE_PP(value) !=<span> IS_NULL) {
    ZVAL_BOOL(&EX_T(opline->result.var).tmp_var, 1<span>);
} else<span> {
    ZVAL_BOOL(&EX_T(opline->result.var).tmp_var, 0<span>);
}</span></span></span></span>
로그인 후 복사

只要value被设置了且不为NULL,isset函数就返回true。

 

小结

这次阅读这两个函数的源码,学习到了:

1、PHP代码在编译期间的执行步骤

2、如何查找PHP语言结构的源码位置

3、如何查找opcode处理函数的具体函数

学无止境,每个人都有自己的短板,只有通过不断学习才能将自己的短板补上。

 

原创文章,文笔有限,才疏学浅,文中若有不正之处,万望告知。

如果本文对你有帮助,请点下推荐吧,谢谢^_^

 

最后再安利一下,我在github有对PHP源码更详细的注解。感兴趣的可以围观一下,给个star。PHP5.4源码注解。可以通过commit记录查看已添加的注解。


参考文章
opcode处理函数查找:http://www.laruence.com/2008/06/18/221.html
PHPopcode深入理解及PHP代码执行步骤:http://www.php-internals.com/book/?p=chapt02/02-03-03-from-opcode-to-handler

 

更多源码文章,欢迎访问个人主页继续查看:hoohack

www.bkjia.comtruehttp://www.bkjia.com/PHPjc/1129313.htmlTechArticle[PHP源码阅读]empty和isset函数,emptyisset 近日被问到PHP中empty和isset函数时怎么判断变量的,刚开始我是一脸懵逼的,因为我自己也只是一知半...
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

인기 기사

R.E.P.O. 에너지 결정과 그들이하는 일 (노란색 크리스탈)
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 최고의 그래픽 설정
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 아무도들을 수없는 경우 오디오를 수정하는 방법
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25 : Myrise에서 모든 것을 잠금 해제하는 방법
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

Ubuntu 및 Debian용 PHP 8.4 설치 및 업그레이드 가이드 Ubuntu 및 Debian용 PHP 8.4 설치 및 업그레이드 가이드 Dec 24, 2024 pm 04:42 PM

PHP 8.4는 상당한 양의 기능 중단 및 제거를 통해 몇 가지 새로운 기능, 보안 개선 및 성능 개선을 제공합니다. 이 가이드에서는 Ubuntu, Debian 또는 해당 파생 제품에서 PHP 8.4를 설치하거나 PHP 8.4로 업그레이드하는 방법을 설명합니다.

CakePHP 데이터베이스 작업 CakePHP 데이터베이스 작업 Sep 10, 2024 pm 05:25 PM

CakePHP에서 데이터베이스 작업은 매우 쉽습니다. 이번 장에서는 CRUD(생성, 읽기, 업데이트, 삭제) 작업을 이해하겠습니다.

CakePHP 날짜 및 시간 CakePHP 날짜 및 시간 Sep 10, 2024 pm 05:27 PM

cakephp4에서 날짜와 시간을 다루기 위해 사용 가능한 FrozenTime 클래스를 활용하겠습니다.

CakePHP 파일 업로드 CakePHP 파일 업로드 Sep 10, 2024 pm 05:27 PM

파일 업로드 작업을 위해 양식 도우미를 사용할 것입니다. 다음은 파일 업로드의 예입니다.

CakePHP 토론 CakePHP 토론 Sep 10, 2024 pm 05:28 PM

CakePHP는 PHP용 오픈 소스 프레임워크입니다. 이는 애플리케이션을 훨씬 쉽게 개발, 배포 및 유지 관리할 수 있도록 하기 위한 것입니다. CakePHP는 강력하고 이해하기 쉬운 MVC와 유사한 아키텍처를 기반으로 합니다. 모델, 뷰 및 컨트롤러 gu

CakePHP 유효성 검사기 만들기 CakePHP 유효성 검사기 만들기 Sep 10, 2024 pm 05:26 PM

컨트롤러에 다음 두 줄을 추가하면 유효성 검사기를 만들 수 있습니다.

CakePHP 로깅 CakePHP 로깅 Sep 10, 2024 pm 05:26 PM

CakePHP에 로그인하는 것은 매우 쉬운 작업입니다. 한 가지 기능만 사용하면 됩니다. cronjob과 같은 백그라운드 프로세스에 대해 오류, 예외, 사용자 활동, 사용자가 취한 조치를 기록할 수 있습니다. CakePHP에 데이터를 기록하는 것은 쉽습니다. log() 함수는 다음과 같습니다.

PHP 개발을 위해 Visual Studio Code(VS Code)를 설정하는 방법 PHP 개발을 위해 Visual Studio Code(VS Code)를 설정하는 방법 Dec 20, 2024 am 11:31 AM

VS Code라고도 알려진 Visual Studio Code는 모든 주요 운영 체제에서 사용할 수 있는 무료 소스 코드 편집기 또는 통합 개발 환경(IDE)입니다. 다양한 프로그래밍 언어에 대한 대규모 확장 모음을 통해 VS Code는

See all articles