深入理解PHP原理之变量(Variables inside PHP)
或許你知道,或許你不知道,PHP是一個(gè)弱類型,動(dòng)態(tài)的腳本語言。所謂弱類型,就是說PHP并不嚴(yán)格驗(yàn)證變量類型(嚴(yán)格來講,PHP是一個(gè)中強(qiáng)類型語言,這部分內(nèi)容會(huì)在以后的文章中敘述),在申明一個(gè)變量的時(shí)候,并不需要顯示指明它保存的數(shù)據(jù)的類型:
<?php$var = 1; //int
$var = "laruence"; //string
$var = 1.0002; //float
$var = array(); // array
$var = new Exception('error'); //object;
動(dòng)態(tài)語言,就是說,PHP的語言結(jié)構(gòu)在運(yùn)行期是可以改變的,比如我們?cè)谶\(yùn)行期require一個(gè)函數(shù)定義文件,從而導(dǎo)致語言的函數(shù)表動(dòng)態(tài)的改變。
所謂腳本語言,就是說,PHP并不是獨(dú)立運(yùn)行的,要運(yùn)行PHP我們需要PHP解析器:
我前面的文章中已經(jīng)講過,PHP的執(zhí)行是通過Zend engine(ZE, Zend引擎), ZE是用C編寫的,大家都知道C是一個(gè)強(qiáng)類型語言,也就是說,在C中所有的變量在它被聲明到最終銷毀,都只能保存一種類型的數(shù)據(jù)。 那么PHP是如何在ZE的基礎(chǔ)上實(shí)現(xiàn)弱類型的呢?
首先要聲明一點(diǎn),如果你以前沒有接觸過PHP的源碼分析,擴(kuò)展開發(fā)。 如果你并不了解PHP的架構(gòu), 沒有聽說ZE,那么我建議你先看看我前面的文章,尤其推薦:- 深入淺出PHP(PHP Internals)
- 深入理解PHP原理之Opcodes
在PHP中,所有的變量都是用一個(gè)結(jié)構(gòu)-zval來保存的, 在Zend/zend.h中我們可以看到zval的定義:
typedef struct _zval_struct {?? zvalue_value value;
?? zend_uint refcount;
?? zend_uchar type;
?? zend_uchar is_ref;
} zval;
其中zvalue_value是真正保存數(shù)據(jù)的關(guān)鍵部分,現(xiàn)在到了揭曉謎底的時(shí)候了,PHP是如何在ZE的基礎(chǔ)上實(shí)現(xiàn)弱類型的呢? 因?yàn)閦value_value是個(gè)聯(lián)合體(union),
typedef union _zvalue_value {?? long lval;
?? double dval;
?? struct {
?? char *val;
?? int len;
?? } str;
?? HashTable *ht;
?? zend_object_value obj;
} zvalue_value;
那么這個(gè)結(jié)構(gòu)是如何儲(chǔ)存PHP中的多種類型的呢?
PHP中常見的變量類型有:
2. 字符串
3. 數(shù)組/關(guān)聯(lián)數(shù)組
4. 對(duì)象
5. 資源
PHP根據(jù)zval中的type字段來儲(chǔ)存一個(gè)變量的真正類型,然后根據(jù)type來選擇如何獲取zvalue_value的值,比如對(duì)于整型和bool值:
zval.type = IS_LONG;//整形?? zval.type = IS_BOOL;//布爾值
就去取zval.value.lval,對(duì)于bool值來說lval∈(0|1);
如果是雙精度,或者float則會(huì)去取zval.value的dval。
而如果是字符串,那么:
這個(gè)時(shí)候,就會(huì)取:
zval.value.str
而這個(gè)也是個(gè)結(jié)構(gòu),存有C分格的字符串和字符串的長(zhǎng)度。
而對(duì)于數(shù)組和對(duì)象,則type分別對(duì)應(yīng)IS_ARRAY, IS_OBJECT, 相對(duì)應(yīng)的則分別取zval.value.ht和obj
比較特別的是資源,在PHP中,資源是個(gè)很特別的變量,任何不屬于PHP內(nèi)建的變量類型的變量,都會(huì)被看作成資源來進(jìn)行保存,比如,數(shù)據(jù)庫(kù)句柄,打開的文件句柄等等。 對(duì)于資源:
type = IS_RESOURCE這個(gè)時(shí)候,會(huì)去取zval.value.lval, 此時(shí)的lval是個(gè)整型的指示器, 然后PHP會(huì)再根據(jù)這個(gè)指示器在PHP內(nèi)建的一個(gè)資源列表中查詢相對(duì)應(yīng)的資源(這部分的內(nèi)容,我以后會(huì)單獨(dú)開一個(gè)篇文章來介紹),目前,你只要知道此時(shí)的 lval就好像是對(duì)應(yīng)于資源鏈表的偏移值。
ZEND_FETCH_RESOURCE(con, type, zval *, default, resource_name, resource_type);借用這樣的機(jī)制,PHP就實(shí)現(xiàn)了弱類型,因?yàn)閷?duì)于ZE的來說,它所面對(duì)的永遠(yuǎn)都是同一種類型,那就是zval。
ps:明天team出去building,我想著應(yīng)該在走之前寫點(diǎn)東西給我的blog reader來消磨周末。今天就簡(jiǎn)單先開個(gè)頭,下一次,我將進(jìn)一步介紹PHP的變量,作用域,以及變量的copy on write和change on write機(jī)制, 待續(xù)….
總結(jié)
以上是生活随笔為你收集整理的深入理解PHP原理之变量(Variables inside PHP)的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 来亦来去难去是什么歌啊
- 下一篇: 深入理解PHP原理之变量作用域