PHP變量的內部表示是如何和用戶腳本中的變量聯系起來的呢?也就是說,如果我在腳本中寫下:
<?php
$var="laruence";
echo $var;
?>
ZE是如何把我的變量var和內部結構zval聯系起來的呢?
深入理解PHP原理之變量中講過,PHP內部都是使用zval來表示變量的,但是對於上面的腳本,我們的變量是有名字的, var。而zval中並沒有相應的字段來體現變量名。
如果你想到了PHP內部一定有一個機制,來實現變量名到zval的映射。那麼你真的是很聰明,;)
在PHP中,所有的變量都會存儲在一個數組中(確切的說是hash table), 並且,PHP也是通過不同的數組來實現變量的作用域的。
當你創建一個變量的時候,PHP會為這個變量分配一個zval,填入相應的變量值,然後將這個變量的名字,和指向這個zval的指針填入一個數組中。然後,當你獲取這個變量的時候,PHP會通過查找這個數組,獲得對應的zval。
查看_zend_executor_globals結構(這個結構在PHP的執行器保存一些執行相關的上下文信息)
struct
_zend_executor_globals
{
....
HashTable*active_symbol_table;/*活動符號表*/
HashTable
symbol_table; /*全局符號表*/
HashTable
included_files;
jmp_buf*bailout;
interror_reporting;
.....
}
其中,全局符號表,保存了在頂層作用域(就是不在任何函數,對象內)的變量。每當調用一個函數(對象的方法)的時候,就會為這個函數創建一個活動符號表,所有在這個函數內定義的變量,都會保存在這個活動符號表中。
對,這就是PHP的變量作用域的實現方式! 舉個列子:
<?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;
?>
在函數sample外面的變量$var,它會被填入全局符號表中,與他對應的有一個zval指針,這個zval保存了一個字符串”I am in the global symbol table”.
函數內的$var, 它會被填入屬於函數sample的活動符號表中,一樣的,與他對應的zval中,保存著字符串”I am in the active symbol table“.
比較特殊的,就是函數sample的參數$para了,這個$para是保存在sample的活動符號表的,但是與他對應的zval指針,會指向一個保存一份全局變量$var的copy的zval(嚴格來講不是copy,是引用,這個涉及到變量的copy on write機制,我會在以後介紹)。
我們都知道PHP對於簡單變量是傳值調用的,但是,我要告訴你的是,PHP並不是簡單的通過復制一個zval來實現傳值的,呵呵,留個懸念,等我下回分解.