摘要:函數(shù)使用格式判斷變量是否為空。對(duì)于,如果變量被如的函數(shù)設(shè)為,則函數(shù)會(huì)返回。,到分支,,返回?cái)?shù)組元素的數(shù)量,為空,因此為,,,因此返回。
近日被問(wèn)到PHP中empty和isset函數(shù)時(shí)怎么判斷變量的,剛開(kāi)始我是一臉懵逼的,因?yàn)槲易约阂仓皇且恢虢?,為了弄懂其真正的原理,趕緊翻開(kāi)源碼研究研究。經(jīng)過(guò)分析可發(fā)現(xiàn)兩個(gè)函數(shù)調(diào)用的都是同一個(gè)函數(shù),因此本文將對(duì)兩個(gè)函數(shù)一起分析。
我在github有對(duì)PHP源碼更詳細(xì)的注解。感興趣的可以圍觀一下,給個(gè)star。PHP5.4源碼注解??梢酝ㄟ^(guò)commit記錄查看已添加的注解。
函數(shù)使用格式 emptybool empty ( mixed $var )
判斷變量是否為空。
issetbool isset ( mixed $var [ , mixed $... ] )
判斷變量是否被設(shè)置且不為NULL。
參數(shù)說(shuō)明對(duì)于empty,在PHP5.5版本以前,empty只支持變量參數(shù),其他類型的參數(shù)會(huì)導(dǎo)致解析錯(cuò)誤,比如函數(shù)調(diào)用的結(jié)果不能作為參數(shù)。
對(duì)于isset,如果變量被如unset的函數(shù)設(shè)為NULL,則函數(shù)會(huì)返回false。如果多個(gè)參數(shù)被傳遞到isset函數(shù),那么只有所有參數(shù)都被設(shè)置isset函數(shù)才會(huì)返回true。從左到右計(jì)算,一旦遇到?jīng)]被設(shè)置的變量就停止。
運(yùn)行示例
$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); // 報(bào)錯(cuò) $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
找到函數(shù)的定義位置
實(shí)際上,empty不是一個(gè)函數(shù),而是一個(gè)語(yǔ)言結(jié)構(gòu)。語(yǔ)言結(jié)構(gòu)是在PHP程序運(yùn)行前編譯好的,因此不能像之前那樣簡(jiǎn)單地搜索PHP_FUNCTION empty或ZEND_FUNCTION empty查看其源碼。要想看empty等語(yǔ)言結(jié)構(gòu)的源碼,先要理解PHP代碼執(zhí)行的機(jī)制。
PHP執(zhí)行代碼會(huì)經(jīng)過(guò)4個(gè)步驟,其流程圖如下所示:
在第一個(gè)階段,即Scanning階段,程序會(huì)掃描zend_language_scanner.l文件將代碼文件轉(zhuǎn)換成語(yǔ)言片段。對(duì)于isset和empty函數(shù)來(lái)說(shuō),在zend_language_scanner.l文件中搜索empty和isset可以得到函數(shù)在此文件中的宏定義如下:
"isset" { return T_ISSET; } "empty" { return T_EMPTY; }
接下來(lái)就到了Parsing階段,這個(gè)階段,程序?qū)_ISSET和T_EMPTY等Tokens轉(zhuǎn)換成有意義的表達(dá)式,此時(shí)會(huì)做語(yǔ)法分析,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函數(shù)最終都執(zhí)行了zend_do_isset_or_isempty函數(shù),在源碼目錄中查找
grep -rn "zend_do_isset_or_isempty"
可以發(fā)現(xiàn),此函數(shù)在zend_compile.c文件中定義。
函數(shù)執(zhí)行步驟
源碼解讀1、解析參數(shù)
2、檢查是否為可寫(xiě)變量
3、如果是變量的op_type是IS_CV(編譯時(shí)期的變量),則設(shè)置其opcode為ZEND_ISSET_ISEMPTY_VAR;否則從active_op_array中獲取下一個(gè)op值,根據(jù)其op值設(shè)置last_op的opcode。
4、設(shè)置了opcode之后,之后會(huì)交給zend_excute執(zhí)行。
IS_CV是編譯器使用的一種cache機(jī)制,這種變量保存著它被引用的變量的地址,當(dāng)一個(gè)變量第一次被引用的時(shí)候,就會(huì)被CV起來(lái),以后這個(gè)變量的引用就不需要再去查找active符號(hào)表了。
對(duì)于empty函數(shù),到了opcode的步驟后,參閱opcode處理函數(shù),可以知道,isset和empty在excute的時(shí)候執(zhí)行的是ZEND_ISSET_ISEMPTY_VAR等一系列函數(shù),以ZEND_ISSET_ISEMPTY_VAR_SPEC_CV_VAR_HANDLER為例,找到這個(gè)函數(shù)的定義在zend_vm_execute.h。查看函數(shù)可以知道,empty函數(shù)的最終執(zhí)行函數(shù)是i_zend_is_true(),而i_zend_is_true函數(shù)定義在zend_execute.h。i_zend_is_true函數(shù)的核心代碼如下:
switch (Z_TYPE_P(op)) { case IS_NULL: result = 0; break; case IS_LONG: case IS_BOOL: case IS_RESOURCE: // empty參數(shù)為整數(shù)時(shí)非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) 是根據(jù)數(shù)組的數(shù)量來(lái)判斷 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; }
這段代碼比較直觀,函數(shù)沒(méi)有對(duì)檢測(cè)值做任何的轉(zhuǎn)換,通過(guò)這段代碼來(lái)進(jìn)一步分析示例中的empty函數(shù)做分析:
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返回?cái)?shù)組元素的數(shù)量,array為空,因此result為0,i_zend_is_true() == 0,!i_zend_is_true() == 1,因此返回true。
empty("0"),到IS_STRING分支,因?yàn)閆_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。
對(duì)于isset函數(shù),最終實(shí)現(xiàn)判斷的代碼是:
if (isset && Z_TYPE_PP(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被設(shè)置了且不為NULL,isset函數(shù)就返回true。
小結(jié)這次閱讀這兩個(gè)函數(shù)的源碼,學(xué)習(xí)到了:
1、PHP代碼在編譯期間的執(zhí)行步驟
2、如何查找PHP語(yǔ)言結(jié)構(gòu)的源碼位置
3、如何查找opcode處理函數(shù)的具體函數(shù)
學(xué)無(wú)止境,每個(gè)人都有自己的短板,只有通過(guò)不斷學(xué)習(xí)才能將自己的短板補(bǔ)上。
原創(chuàng)文章,文筆有限,才疏學(xué)淺,文中若有不正之處,萬(wàn)望告知。
如果本文對(duì)你有幫助,請(qǐng)點(diǎn)下推薦吧,謝謝^_^
我在github有對(duì)PHP源碼更詳細(xì)的注解。感興趣的可以圍觀一下,給個(gè)star。PHP5.4源碼注解??梢酝ㄟ^(guò)commit記錄查看已添加的注解。
參考文章
opcode處理函數(shù)查找:http://www.laruence.com/2008/06/18/221.html
PHPopcode深入理解及PHP代碼執(zhí)行步驟:http://www.php-internals.com/book/?p=chapt02/02-03-03-from-opcode-to-handler
更多源碼文章,歡迎訪問(wèn)個(gè)人主頁(yè)繼續(xù)查看:hoohack
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/21596.html
摘要:如果還有人問(wèn)你兩者區(qū)別,馬上甩出這種圖有興趣可以往下閱讀,官方手冊(cè)給出的例子手冊(cè)這是一張將區(qū)別的表格,從表格中我們可以發(fā)現(xiàn)返回值等同返回值等同。 簡(jiǎn)單談一下isset和empty的區(qū)別? 如果你是在面試,碰巧面試官提了這個(gè)問(wèn)題。你可以這樣回答: 如果變量值為0、空字符串、空數(shù)組等等,empty認(rèn)為它是空的,而isset認(rèn)為它不是空的。 如果變量不存在,isset和empty都認(rèn)為它是...
摘要:定義先來(lái)看下兩個(gè)函數(shù)的手冊(cè)檢查一個(gè)變量是否為空判斷一個(gè)變量是否被認(rèn)為是空的。當(dāng)一個(gè)變量并不存在,或者它的值等同于,那么它會(huì)被認(rèn)為不存在。如果變量不存在的話,并不會(huì)產(chǎn)生警告。檢測(cè)變量是否已設(shè)置并且非檢測(cè)變量是否設(shè)置,并且不是。 導(dǎo)語(yǔ) 老生常談的話題,之所以用這兩個(gè)函數(shù)做對(duì)比,是因?yàn)槌S脕?lái)判斷變量是否為空,但是它們有些不同,下面進(jìn)行下對(duì)比。 定義 先來(lái)看下兩個(gè)函數(shù)的手冊(cè) empty emp...
摘要:支持基于段方法和查詢字符串方法兩種形式的。里的方法就是利用類來(lái)實(shí)現(xiàn)解析出類名方法名。在類的構(gòu)造函數(shù)里有一步方法代碼如下如果你的原始是的話,經(jīng)過(guò)這個(gè)方法處理,你會(huì)得到參考文章 路由的目的是為了從URL中解析出class類名是什么,method方法名是什么,所傳的參數(shù)有哪些,參數(shù)值又是什么,類文件存在的路徑是哪。最終實(shí)現(xiàn)方法的調(diào)度。 CI支持基于段方法和查詢字符串方法兩種形式的URL。 基...
摘要:重載在中就大量應(yīng)用了重載相關(guān)知識(shí),如在中就用到了方法重載知識(shí)使用魔術(shù)方法來(lái)動(dòng)態(tài)創(chuàng)建類中未定義或不可見(jiàn)的靜態(tài)方法。中通過(guò)引入魔術(shù)方法來(lái)實(shí)現(xiàn)動(dòng)態(tài)的創(chuàng)建類屬性和方法,包括屬性重載的魔術(shù)方法和方法重載的魔術(shù)方法。 說(shuō)明:本文主要講述PHP中重載概念,由于Laravel框架中經(jīng)常使用這塊知識(shí)點(diǎn),并且PHP的重載概念又與其他OOP語(yǔ)言如JAVA中重載概念不一樣,故復(fù)習(xí)并記錄相關(guān)知識(shí)點(diǎn)。同時(shí),作者會(huì)...
摘要:為什么變量的值為字符串,但同時(shí)會(huì)是空值呢讓我們?cè)谧兞可蠂L試使用其它一些函數(shù)來(lái)進(jìn)行判斷吧以上結(jié)果為譯者注這邊的結(jié)果可能存在問(wèn)題的結(jié)果同樣為,可以到這里去運(yùn)行下查看結(jié)果。和函數(shù)執(zhí)行結(jié)果符合預(yù)期判斷,唯獨(dú)函數(shù)返回了錯(cuò)誤結(jié)果。 本文首發(fā)于 震驚 php empty 函數(shù)判斷結(jié)果為空,但實(shí)際值卻為非空,轉(zhuǎn)載請(qǐng)注明出處。 最近我在一個(gè)項(xiàng)目中使用 empty 時(shí)獲取到了一些意料之外的結(jié)果。下面是我處...
閱讀 1550·2021-11-24 09:38
閱讀 3396·2021-11-18 10:02
閱讀 3289·2021-09-22 15:29
閱讀 2974·2021-09-22 15:15
閱讀 1079·2021-09-13 10:25
閱讀 1893·2021-08-17 10:13
閱讀 2042·2021-08-04 11:13
閱讀 2005·2019-08-30 15:54