前面的文章(深入理解PHP原理之變量(Variables inside PHP))介紹了PHP變量的內(nèi)部表示,但是,問(wèn)題是,這些內(nèi)部表示是如何和用戶腳本中的變量聯(lián)系起來(lái)的呢?也就是說(shuō),如果我在腳本中寫下:
<?php $var = "laruence"; echo $var; ?>
ZE是如何把我的變量var和內(nèi)部結(jié)構(gòu)zval聯(lián)系起來(lái)的呢?
深入理解PHP原理之變量中講過(guò),PHP內(nèi)部都是使用zval來(lái)表示變量的,但是對(duì)于上面的腳本,我們的變量是有名字的, var。而zval中并沒(méi)有相應(yīng)的字段來(lái)體現(xiàn)變量名。
如果你想到了PHP內(nèi)部一定有一個(gè)機(jī)制,來(lái)實(shí)現(xiàn)變量名到zval的映射。那么你真的是很聰明,;)
推薦學(xué)習(xí):PHP視頻教程
在PHP中,所有的變量都會(huì)存儲(chǔ)在一個(gè)數(shù)組中(確切的說(shuō)是hash table), 并且,PHP也是通過(guò)不同的數(shù)組來(lái)實(shí)現(xiàn)變量的作用域的。
當(dāng)你創(chuàng)建一個(gè)變量的時(shí)候,PHP會(huì)為這個(gè)變量分配一個(gè)zval,填入相應(yīng)的變量值,然后將這個(gè)變量的名字,和指向這個(gè)zval的指針填入一個(gè)數(shù)組中。然后,當(dāng)你獲取這個(gè)變量的時(shí)候,PHP會(huì)通過(guò)查找這個(gè)數(shù)組,獲得對(duì)應(yīng)的zval。
查看_zend_executor_globals結(jié)構(gòu)(這個(gè)結(jié)構(gòu)在PHP的執(zhí)行器保存一些執(zhí)行相關(guān)的上下文信息)
struct _zend_executor_globals { .... HashTable *active_symbol_table;/*活動(dòng)符號(hào)表*/ HashTable symbol_table; /*全局符號(hào)表*/ HashTable included_files; jmp_buf *bailout; int error_reporting; ..... }
其中,全局符號(hào)表,保存了在頂層作用域(就是不在任何函數(shù),對(duì)象內(nèi))的變量。每當(dāng)調(diào)用一個(gè)函數(shù)(對(duì)象的方法)的時(shí)候,就會(huì)為這個(gè)函數(shù)創(chuàng)建一個(gè)活動(dòng)符號(hào)表,所有在這個(gè)函數(shù)內(nèi)定義的變量,都會(huì)保存在這個(gè)活動(dòng)符號(hào)表中。
對(duì),這就是PHP的變量作用域的實(shí)現(xiàn)方式! 舉個(gè)列子:
<?php $var = "I am in the global symbol table"; function sample($para){ $var = "I am in the active symbol table"; echo $var; } sample($var); echo $var; ?>
在函數(shù)sample外面的變量$var,它會(huì)被填入全局符號(hào)表中,與他對(duì)應(yīng)的有一個(gè)zval指針,這個(gè)zval保存了一個(gè)字符串”I am in the global symbol table”.
函數(shù)內(nèi)的$var, 它會(huì)被填入屬于函數(shù)sample的活動(dòng)符號(hào)表中,一樣的,與他對(duì)應(yīng)的zval中,保存著字符串”I am in the active symbol table“.
比較特殊的,就是函數(shù)sample的參數(shù)$para了,這個(gè)$para是保存在sample的活動(dòng)符號(hào)表的,但是與他對(duì)應(yīng)的zval指針,會(huì)指向一個(gè)保存一份全局變量$var的copy的zval(嚴(yán)格來(lái)講不是copy,是引用)。