PHP源碼分析之變數的預存程序分解_php執行個體

來源:互聯網
上載者:User

PHP代碼如下:

複製代碼 代碼如下:
$php_var = 1; 

對應C的代碼是:

複製代碼 代碼如下:
zval* c_var;    //定義PHP變數指標 
MAKE_STD_ZVAL(c_var);  //初始化PHP變數 
ZVAL_LONG(c_var,1) ;//賦值 
ZEND_SET_SYMBL( EG(active_symbol_table), " php_var ", c_var);//註冊到全域變數符號表

一.首先看第一行: zval* c_var;//申明一個zval指標c_var; zval的結構如下:

複製代碼 代碼如下:

struct _zval_struct { 
    /* Variable information */ 
    zvalue_value value;     /* 變數的值 */ 
    zend_uint refcount;     /* 引用計數,記憶體回收的時候用到 */ 
    zend_uchar type;        /* 變數類型 */ 
    zend_uchar is_ref;      /* 是否為引用變數 */ 
}; 
typedef struct _zval_struct zval; 

其中值zvalue_value的結構如下:

複製代碼 代碼如下:

typedef union _zvalue_value { 
    long lval;              /* 長整形*/ 
    double dval;            /* 雙精確度類型 */ 
    struct {                  /* 字串類型的值 */ 
        char *val;             
        int len; 
    } str; 
    HashTable *ht;              /* 數群組類型的值 */ 
    zend_object_value obj;     /*物件類型的值*/ 
} zvalue_value; 

二.接下來看第二行: MAKE_STD_ZVAL(new_val);//變數初始化 相關宏如下: //初始化

複製代碼 代碼如下:

#define MAKE_STD_ZVAL(zv)                \ 
    ALLOC_ZVAL(zv); \ 
    INIT_PZVAL(zv); 
 
#define ALLOC_ZVAL(z)   \ 
    ZEND_FAST_ALLOC(z, zval, ZVAL_CACHE_LIST) 
 
#define ZEND_FAST_ALLOC(p, type, fc_type)   \ 
    (p) = (type *) emalloc(sizeof(type)) 
 
#define INIT_PZVAL(z)       \ 
    (z)->refcount = 1;      \ 
    (z)->is_ref = 0; 


展開後為:

複製代碼 代碼如下:

(c_var) = (zval *) emalloc(sizeof(zval));  //分配記憶體 
(c_var)-> refcount = 1;  //引用計數初始化 
(c_var)-> is_ref = 0; //是否引用 

可以看到其作用就是分配記憶體,初始化refcount,is_ref

三.下面看第三行 ZVAL_LONG(c_var,1) 相關宏為:

複製代碼 代碼如下:

//定義值 
#define ZVAL_LONG(z, l) {           \ 
     Z_TYPE_P(z) = IS_LONG;      \ 
     Z_LVAL_P(z) = l;            \ 

#define Z_TYPE_P(zval_p)    Z_TYPE(*zval_p) 
#define Z_TYPE(zval)        (zval).type 
#define Z_LVAL_P(zval_p)    Z_LVAL(*zval_p) 
#define Z_LVAL(zval)            (zval).value.lval 

展開後為:

複製代碼 代碼如下:

(* c_var).type = IS_LONG; 
(* c_var).value = 1; 

四:接下來看第四行: ZEND_SET_SYMBOL( EG(active_symbol_table), “php_var”, c_var); 首先說明下PHP的變數是存在一個hashtable裡的

複製代碼 代碼如下:

struct _zend_executor_globals {   
        …. 
        HashTable symbol_table;//全域變數的符號表   
        HashTable *active_symbol_table;//局部變數的符號表   
        ….. 
    };   

Hashtable的Key為變數的名稱,即php_var,值為指向PHP變數的指標,即c_var指標; 相關宏為:

複製代碼 代碼如下:

#define ZEND_SET_SYMBOL(symtable, name, var)          \   {                                                     \ 
        char *_name = (name);                         \ 
        ZEND_SET_SYMBOL_WITH_LENGTH(symtable, _name, strlen(_name)+1, var, 1, 0);   \ 

//主要的實現為下面這個函數: 
#define ZEND_SET_SYMBOL_WITH_LENGTH(symtable, name, name_length, var, _refcount, _is_ref)                                                       \ 
    {                                                                        
        zval **orig_var;                                        \  
        if (zend_hash_find(symtable, (name), (name_length), (void **) &orig_var)==SUCCESS                                                         \ 
            && PZVAL_IS_REF(*orig_var)) {                     \ 
            (var)->refcount = (*orig_var)->refcount;                  \ 
            (var)->is_ref = 1;                                \ 
            if (_refcount) {                                      \ 
                (var)->refcount += _refcount-1;               \ 
            }                                             \ 
            zval_dtor(*orig_var);                             \ 
            **orig_var = *(var);                                  \ 
            FREE_ZVAL(var);                               \ 
        } else {                                              \ 
            (var)->is_ref = _is_ref;                              \ 
            if (_refcount) {                                      \ 
                (var)->refcount = _refcount;                      \ 
            }                                             \ 
            zend_hash_update(symtable, (name), (name_length), &(var), sizeof(zval *), NULL);                                                           \ 
        }                                                  \ 
    }            

該函數的功能是:
1. 如果全域符號表已經存在該變數且是參考型別,則

a. 將原來變數的引用計數refcount,is_ref資訊賦給c_var;
b. 釋放掉原來變數zvalue的值,比如原來其值指向的是一個mysql串連資源,則釋放該資源。
c. 將c_var指向的變數賦值給原來的變數 d. 釋放c_var的記憶體空間 這樣保證了,如果變數被應用,值一起改變。比如如果前面有$b=&a;

2. 如果全域符號表不存在該變數或者存在該變數但不是引用變數,則直接改變其值。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.