国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院

首頁 > 編程 > 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編程

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

  • 發表評論 共有條評論
    用戶名: 密碼:
    驗證碼: 匿名發表
    国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院
    国产精品白浆视频免费观看| 国产成人精品实拍在线| 国产激情视频在线| 五月婷婷丁香激情| 亚洲欧美综合乱码精品成人网| 91在线看片| 亚洲成人av高清| gogogogo高清视频在线| eeuss影院在线| 福利在线观看| 国产视频1区| 亚洲国产aⅴ精品| 久久香蕉av| 国产对白国语对白| 亚洲夜夜综合| 国产女人伦码一区二区三区不卡| 欧美日韩一区二区三区视视频 | 羞羞视频在线免费看| 亚洲精品乱码电影在线观看| 福利视频网址导航| 国产精品入口免费麻豆| 精品日韩av| 美女免费视频黄| 国产精品入口麻豆免费| 欧洲有码在线视频| 国产九色在线| 色中文字幕在线| 国产日韩欧美一区二区三区视频| 色悠久久久久综合网小说| 国产蜜臀av在线播放| 国产激情自拍视频| 高潮白浆视频| 午夜影院在线| 国产成免费视频| 国产一二三区在线视频| 激情丁香久久| 啪啪免费视频一区| 中文产幕区在线观看| 国产偷窥洗澡视频| 最新黄网在线观看| 国产在线中文字幕| 超碰人人在线| 超碰在线网址| 国产免费黄网站| 九九在线观看免费视频| 黄网在线免费| 九九热视频免费观看| 亚洲天堂电影在线观看| 国产无遮挡又黄又爽免费软件| 懂色av中文在线| 黄色一级视频网站| 青青青手机在线视频观看| h网站久久久| xxxxx中文字幕| www.亚洲视频| 青青久在线视频免费观看| sese一区| 尤物视频网站在线观看| 国产麻豆免费| 国产一级在线| 2021av在线| 午夜影院免费看| www狠狠操| 国产乱xxⅹxx国语对白| 国产精品久久人| 在线视频婷婷| 国产免费黄视频在线观看| 国产精品bbw一区二区三区| 在线中文字幕资源| 国产免费自拍视频| 在线黄色.com| 国产永久av在线| 国产二区三区在线| 日本电影全部在线观看网站视频 | 天堂在线视频| 99热在线免费播放| 69久久精品| 黄色国产网站在线观看| 国产精品99爱免费视频| 日本电影在线观看| 国产黄色大片在线观看| 丁香视频五月| 国产福利小视频在线| 麻豆精品传媒视频观看| 在线天堂中文| 国产精品美女视频免费观看软件| www.av在线视频| 99福利在线| 国产黄色在线| 国产香蕉免费精品视频| 女同一区二区免费aⅴ| 中文字幕av网| 999国产在线视频| 国产网站免费看| 99热在线观看免费| 国产乱子视频| 中文字幕视频在线| 国产免费高清| 国产香蕉视频在线看| 91福利在线免费| 国产麻豆精品高清在线播放| 久久一本精品| 成人无遮挡免费网站视频在线观看| 欧美亚洲系列| www.色婷婷| 欧美韩日国产| 激情综合丁香| 91福利在线免费| 国产一区二区三区福利| 国产视频资源| √天堂资源中文www| 精品偷拍激情视频在线观看| 国产精品剧情一区二区在线观看| 国产精品扒开做爽爽爽的视频| 国产午夜精品一区理论片| 国产欧美日韩专区| 伊人国产在线看一| 欧美精品久久久久久久小说| 日本精品一区二区三区在线播放| 美女av在线播放| 国产女人在线视频| 国产黄色免费看| eeuss影院www在线观看| 国产秒拍福利视频露脸| 国产成人久久精品77777| a视频在线观看免费| 国产字幕在线看| 国产福利在线播放| 九九热在线视频观看| 国产黄a三级三级三级av在线看 | 国产丝袜自拍| 依依成人在线| 国产原创av在线| 最近最好的中文字幕2019免费 | 国产视频1区| www免费在线观看| 国产私人影院| 国产在线观看色| 伊人电影在线观看| 国产特级淫片免费看| 日本动漫同人动漫在线观看| 国产精品久久麻豆| 超碰97在线免费观看| 国产黄色在线观看| 黄网址在线永久免费观看| 国产经典av| 国产色婷婷在线| 九九热视频在线观看| av文字幕在线观看| 国产黄色av免费看| 国产麻豆精品一区二区三区v视界| 亚洲综合天堂网| 97视频网站| 国产精品186在线观看在线播放| 狠狠干在线视频| 久热久精久品这里在线观看| 在线中文视频| 国产福利电影在线观看| 天堂资源最新版在线视频观看免费网| www.久草.com| 青青草观看免费视频在线| eeuss影院www在线观看| 99re6在线视频精品免费| 91精品大全| 精品视频一二区| 精品极品三级久久久久| 国产精品国产国产aⅴ| 午夜视频在线看| 中文字幕久热在线精品| 最近中文字幕mv2018在线高清| 国产精品午夜久久久久久| 国产麻豆综合视频在线观看| 久久五月精品| 国产天堂在线| 亚洲精品一区中文字幕电影| 天天操夜夜做| 国产鲁鲁视频在线观看特色| 国产精品入口麻豆免费看| 在线一区二区三区精品| gogo在线高清视频| 粉嫩av一区| 在线三级av| 麻豆精品不卡国产免费看| 99热最新网址| 一级二级三级在线观看| 国产中文字幕在线看| 国产寡妇树林野战在线播放| 99色在线观看| 嫩草在线播放| 国产成人夜间影院在线观看| 亚洲91av| 五月综合网站| 一区二区三区四区在线免费视频| 日本片在线看| 国产福利热线视频| 欧美精品久久久久久久小说| 国产视频三级在线观看播放| 日韩在线天堂| 中文视频在线| 国产高清免费av在线|