復制代碼 代碼如下:$php_var = 1; 對應C的代碼是:復制代碼 代碼如下:zval* c_var; //定義PHP變量指針
MAKE_STD_ZVAL(c_var); //初始化PHP變量
ZVAL_LONG(c_var,1) ;//賦值
ZEND_SET_SYMBL( EG " /> 蜜桃狠狠色伊人亚洲综合网站,少妇两个奶头喷出奶水了怎么办,秋霞伦理电影在线看

天天躁日日躁狠狠躁AV麻豆-天天躁人人躁人人躁狂躁-天天澡夜夜澡人人澡-天天影视香色欲综合网-国产成人女人在线视频观看-国产成人女人视频在线观看

PHP源碼分析之變量的存儲過程分解

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. 如果全局符號表不存在該變量或者存在該變量但不是引用變量,則直接改變其值。

php技術PHP源碼分析之變量的存儲過程分解,轉載需保留來源!

鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。

主站蜘蛛池模板: 99久久免费精品国产 | 中文字幕无码他人妻味 | 4399的视频BD高清在线观看免费 | 亚洲精品福利一区二区在线观看 | 99久久久无码国产精精品 | 亚洲免费黄色片 | 兽皇VIDEO另类HD | 狠狠色丁香婷婷久久综合五月 | 日日天干夜夜狠狠爱 | 老司机福利在视频在ae8 | 999久久国产精品免费人妻 | 亚洲午夜一区二区电影院 | 最新精品学生国产自在现拍 | 成人无码在线视频区 | 亚洲一区在线观看无码欧美 | 深夜释放自己污在线看 | 美女岔开腿露出粉嫩花苞 | 高清国语自产拍在线 | 99re热精品视频国产免费 | 亚洲中文无码AV在线观看 | 国产呦精品一区二区三区下载 | 久久亚洲电影www电影网 | 精品淑女少妇AV久久免费 | 日韩亚洲欧美中文在线 | 小莹的性荡生活45章 | 欧美精品成人一区二区在线观看 | 国产AV精品久久久毛片 | 女同给老师下媚药 | 挠黑色超薄丝袜脚心vk40分钟 | 精品国产三级a | 久久精品国产96精品亚洲 | 一本道dvd久久综合高清免费 | 99视频免费看| 黄梅戏mp3大全 | 国产揄拍国产精品 | 嫩草影院在线观看精品 | 久久国产免费一区二区三区 | 欧美深深色噜噜狠狠yyy | 黄色三级视频在线观看 | 91亚洲 欧美 国产 制服 动漫 | 欧美男男网站免费观看videos |