亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > PHP > 正文

[PHP源碼閱讀]empty和isset函數

2020-03-22 20:12:29
字體:
來源:轉載
供稿:網友
  • 近日被問到PHP中empty和isset函數時怎么判斷變量的,剛開始我是一臉懵逼的,因為我自己也只是一知半解,為了弄懂其真正的原理,趕緊翻開源碼研究研究。經過分析可發現兩個函數調用的都是同一個函數,因此本文將對兩個函數一起分析。

    我在github有對PHP源碼更詳細的注解。感興趣的可以圍觀一下,給個star。PHP5.4源碼注解??梢酝ㄟ^commit記錄查看已添加的注解。

    函數使用格式empty

    bool empty ( mixed $var )


    判斷變量是否為空。

    isset

    bool isset ( mixed $var [ , mixed $... ] )

    判斷變量是否被設置且不為NULL。

    參數說明

    對于empty,在PHP5.5版本以前,empty只支持變量參數,其他類型的參數會導致解析錯誤,比如函數調用的結果不能作為參數。

    對于isset,如果變量被如unset的函數設為NULL,則函數會返回false。如果多個參數被傳遞到isset函數,那么只有所有參數都被設置isset函數才會返回true。從左到右計算,一旦遇到沒被設置的變量就停止。

    運行示例
    $result = empty(0); // true$result = empty(null); // true$result = empty(false); // true$result = empty(array()); // true$result = empty('0'); // true$result = empty(1); // false$result = empty(callback function); // 報錯$a = null;$result = isset($a); // false;$a = 1;$result = isset($a); // true;$a = 1;$b = 2;$c = 3;$result = isset($a, $b, $c); // true
    $a = 1;$b = null;$c = 3;$result = isset($a, $b, $c); // false
    找到函數的定義位置

    實際上,empty不是一個函數,而是一個語言結構。語言結構是在PHP程序運行前編譯好的,因此不能像之前那樣簡單地搜索'PHP_FUNCTION empty'或'ZEND_FUNCTION empty'查看其源碼。要想看empty等語言結構的源碼,先要理解PHP代碼執行的機制。

    PHP執行代碼會經過4個步驟,其流程圖如下所示:


    在第一個階段,即Scanning階段,程序會掃描zend_language_scanner.l文件將代碼文件轉換成語言片段。對于isset和empty函數來說,在zend_language_scanner.l文件中搜索empty和isset可以得到函數在此文件中的宏定義如下:

    <ST_IN_SCRIPTING>'isset' {return T_ISSET;}<ST_IN_SCRIPTING>'empty' {return T_EMPTY;}


    接下來就到了Parsing階段,這個階段,程序將T_ISSET和T_EMPTY等Tokens轉換成有意義的表達式,此時會做語法分析,Tokens的yacc保存在zend_language_parser.y文件中,可以找到T_ISSET和T_EMPTY的定義:

    internal_functions_in_yacc:T_ISSET '(' isset_variables ')' { $$ = $3; }| T_EMPTY '(' variable ')' { zend_do_isset_or_isempty(ZEND_ISEMPTY, &$$, &$3 TSRMLS_CC); }| T_INCLUDE expr { zend_do_include_or_eval(ZEND_INCLUDE, &$$, &$2 TSRMLS_CC); }| T_INCLUDE_ONCE expr { zend_do_include_or_eval(ZEND_INCLUDE_ONCE, &$$, &$2 TSRMLS_CC); }| T_EVAL '(' expr ')' { zend_do_include_or_eval(ZEND_EVAL, &$$, &$3 TSRMLS_CC); }| T_REQUIRE expr { zend_do_include_or_eval(ZEND_REQUIRE, &$$, &$2 TSRMLS_CC); }| T_REQUIRE_ONCE expr { zend_do_include_or_eval(ZEND_REQUIRE_ONCE, &$$, &$2 TSRMLS_CC); };


    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函數的核心代碼如下:

            switch (Z_TYPE_P(op)) {        case IS_NULL:            result = 0;            break;        case IS_LONG:        case IS_BOOL:        case IS_RESOURCE:            // empty參數為整數時非0的話就為false            result = (Z_LVAL_P(op)?1:0);            break;        case IS_DOUBLE:            result = (Z_DVAL_P(op) ? 1 : 0);            break;        case IS_STRING:            if (Z_STRLEN_P(op) == 0                || (Z_STRLEN_P(op)==1 && Z_STRVAL_P(op)[0]=='0')) {                // empty('0') == true                result = 0;            } else {                result = 1;            }            break;        case IS_ARRAY:            // empty(array) 是根據數組的數量來判斷            result = (zend_hash_num_elements(Z_ARRVAL_P(op))?1:0);            break;        case IS_OBJECT:            if(IS_ZEND_STD_OBJECT(*op)) {                TSRMLS_FETCH();                if (Z_OBJ_HT_P(op)->cast_object) {                    zval tmp;                    if (Z_OBJ_HT_P(op)->cast_object(op, &tmp, IS_BOOL TSRMLS_CC) == SUCCESS) {                        result = Z_LVAL(tmp);                        break;                    }                } else if (Z_OBJ_HT_P(op)->get) {                    zval *tmp = Z_OBJ_HT_P(op)->get(op TSRMLS_CC);                    if(Z_TYPE_P(tmp) != IS_OBJECT) {                        /* for safety - avoid loop */                        convert_to_boolean(tmp);                        result = Z_LVAL_P(tmp);                        zval_ptr_dtor(&tmp);                        break;                    }                }            }            result = 1;            break;        default:            result = 0;            break;    }


    這段代碼比較直觀,函數沒有對檢測值做任何的轉換,通過這段代碼來進一步分析示例中的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(html' target='_blank'>value) != IS_NULL) {    ZVAL_BOOL(&EX_T(opline->result.var).tmp_var, 1);} else {    ZVAL_BOOL(&EX_T(opline->result.var).tmp_var, 0);}

    只要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

    PHP編程

    鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。

  • 發表評論 共有條評論
    用戶名: 密碼:
    驗證碼: 匿名發表
    亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
    在线午夜精品自拍| 在线视频欧美性高潮| 日韩久久精品成人| 精品国内亚洲在观看18黄| 爽爽爽爽爽爽爽成人免费观看| 亚洲欧美福利视频| 亚洲精品91美女久久久久久久| 国产一区二区香蕉| 欧美精品久久久久a| 成人国内精品久久久久一区| 成人久久一区二区| 亚洲第一国产精品| 成人免费淫片aa视频免费| 久久精品国产综合| 欧美国产极速在线| 中文字幕日韩欧美在线视频| 欧美性感美女h网站在线观看免费| 久久久久国产视频| 第一福利永久视频精品| 欧美精品xxx| 久久久久久国产精品美女| 欧美尤物巨大精品爽| 国语自产精品视频在线看| 亚洲性线免费观看视频成熟| 91精品国产91久久久久久不卡| 国产精品精品视频一区二区三区| 久久人体大胆视频| 欧美激情精品久久久久久| 国产精品爽黄69天堂a| 日韩精品久久久久久久玫瑰园| 欧美性资源免费| 91wwwcom在线观看| 欧美xxxx14xxxxx性爽| 亚洲护士老师的毛茸茸最新章节| 亚洲免费人成在线视频观看| 欧美国产中文字幕| 亚洲女人被黑人巨大进入| 国产精品色午夜在线观看| 欧美日韩激情小视频| 日韩av中文字幕在线免费观看| 国产一区二区动漫| 在线精品高清中文字幕| 亚洲欧美成人一区二区在线电影| 亚洲男人的天堂在线| 亚洲综合视频1区| 国产成人免费av| 欧美一级高清免费| 欧美激情网友自拍| 中文字幕欧美日韩精品| 欧美猛男性生活免费| 精品视频—区二区三区免费| 羞羞色国产精品| 久久99视频免费| 日韩美女视频中文字幕| 国产精品久久久久91| www.日韩欧美| 欧美放荡办公室videos4k| 久久久久久久久久久国产| 日韩精品福利网站| 日韩视频第一页| 久久久99免费视频| 国内精品伊人久久| www.日韩视频| 精品久久久久久久大神国产| 日本19禁啪啪免费观看www| 欧美日韩在线免费| 一区二区欧美日韩视频| 亚洲精品美女久久久久| 日韩欧美中文在线| 国产一区二区三区在线观看网站| 国产亚洲欧美视频| 精品国产乱码久久久久酒店| 久久国产精品首页| 欧美精品videos| 国产伦精品免费视频| 久久久极品av| 欧美日韩福利在线观看| 欧美亚洲另类制服自拍| 国产精品444| 亚洲最大激情中文字幕| 国产91色在线|免| 欧美一区二区三区艳史| 久久躁日日躁aaaaxxxx| 亚洲人成五月天| 欧美疯狂做受xxxx高潮| 欧美性猛xxx| 精品国产一区二区三区在线观看| 日韩电影中文字幕av| 国产精品久久99久久| 亚洲国产精品久久久久| 国产美女直播视频一区| 精品欧美激情精品一区| 韩国美女主播一区| 久久噜噜噜精品国产亚洲综合| 1769国产精品| 久久久综合免费视频| 久久精品国产91精品亚洲| 中文字幕精品在线视频| 91精品国产自产在线观看永久| 国产成人精品亚洲精品| 久久精品中文字幕免费mv| 久久九九热免费视频| 亚洲人成网站777色婷婷| 中文字幕欧美专区| 性色av一区二区三区免费| 欧美精品videos| 色婷婷av一区二区三区久久| 国内精品视频久久| 欧美激情性做爰免费视频| 色噜噜狠狠色综合网图区| 91精品久久久久久久久久另类| 性欧美暴力猛交69hd| 理论片在线不卡免费观看| 亚洲一区二区三区四区在线播放| 免费不卡在线观看av| 亚洲国产成人在线视频| 亚洲九九九在线观看| 国产热re99久久6国产精品| 日韩精品在线观| 亚洲成人三级在线| 久久伊人精品一区二区三区| 日韩一区二区精品视频| 亚洲精品小视频在线观看| 欧美夜福利tv在线| 亚洲欧洲第一视频| 亚洲一区二区自拍| 91免费高清视频| 国产成人精品午夜| 欧美亚洲另类激情另类| 国产精品高潮呻吟久久av黑人| 国产一区二区三区在线免费观看| 久久久久www| 成人乱人伦精品视频在线观看| 国产精品久久999| 欧美成人午夜激情视频| 成人综合国产精品| 最近2019年日本中文免费字幕| 国产欧美在线视频| 欧美亚洲国产精品| 亚洲无限乱码一二三四麻| 亚洲福利视频在线| 96sao精品视频在线观看| 亚洲精美色品网站| 国产精品久久久久91| 欧美精品videos另类日本| 中文字幕免费精品一区| 亚洲天堂免费观看| 欧美在线视频在线播放完整版免费观看| 久久久久久久国产精品| 亚洲精品综合精品自拍| 亚洲aa在线观看| 欧美激情视频网| 成人国产精品日本在线| 国产美女被下药99| 国产欧美一区二区三区久久| 亚洲色图校园春色| 欧美精品一二区| 日韩免费观看视频| 91成人国产在线观看| 高清一区二区三区四区五区| 欧洲s码亚洲m码精品一区| 亚洲毛片在线观看| 亚洲乱码国产乱码精品精天堂| 国产主播喷水一区二区|