PHP源碼分析之變量的存儲過程分解
PHP代碼如下:
對應(yīng)C的代碼是:
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;//申明一個(gè)zval指針c_var; zval的結(jié)構(gòu)如下:
struct _zval_struct {
/* Variable information */
zvalue_value value; /* 變量的值 */
zend_uint refcount; /* 引用計(jì)數(shù),垃圾回收的時(shí)候用到 */
zend_uchar type; /* 變量類型 */
zend_uchar is_ref; /* 是否為引用變量 */
};
typedef struct _zval_struct zval;
其中值zvalue_value的結(jié)構(gòu)如下:
typedef union _zvalue_value {
long lval; /* 長整形*/
double dval; /* 雙精度類型 */
struct { /* 字符串類型的值 */
char *val;
int len;
} str;
HashTable *ht; /* 數(shù)組類型的值 */
zend_object_value obj; /*對象類型的值*/
} zvalue_value;
二.接下來看第二行: MAKE_STD_ZVAL(new_val);//變量初始化 相關(guān)宏如下: //初始化
#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)); //分配內(nèi)存
(c_var)-> refcount = 1; //引用計(jì)數(shù)初始化
(c_var)-> is_ref = 0; //是否引用
可以看到其作用就是分配內(nèi)存,初始化refcount,is_ref
三.下面看第三行 ZVAL_LONG(c_var,1) 相關(guān)宏為:
//定義值
#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的變量是存在一個(gè)hashtable里的
struct _zend_executor_globals {
….
HashTable symbol_table;//全局變量的符號表
HashTable *active_symbol_table;//局部變量的符號表
…..
};
Hashtable的Key為變量的名稱,即php_var,值為指向PHP變量的指針,即c_var指針; 相關(guān)宏為:
#define ZEND_SET_SYMBOL(symtable, name, var) \ { \
char *_name = (name); \
ZEND_SET_SYMBOL_WITH_LENGTH(symtable, _name, strlen(_name)+1, var, 1, 0); \
}
//主要的實(shí)現(xiàn)為下面這個(gè)函數(shù):
#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); \
} \
}
該函數(shù)的功能是:
1. 如果全局符號表已經(jīng)存在該變量且是引用類型,則
a. 將原來變量的引用計(jì)數(shù)refcount,is_ref信息賦給c_var;
b. 釋放掉原來變量zvalue的值,比如原來其值指向的是一個(gè)mysql連接資源,則釋放該資源。
c. 將c_var指向的變量賦值給原來的變量 d. 釋放c_var的內(nèi)存空間 這樣保證了,如果變量被應(yīng)用,值一起改變。比如如果前面有$b=&a;
2. 如果全局符號表不存在該變量或者存在該變量但不是引用變量,則直接改變其值。
- php調(diào)用mysql存儲過程實(shí)例分析
- PHP調(diào)用MySQL存儲過程并返回值的方法
- PHP調(diào)用MsSQL Server 2012存儲過程獲取多結(jié)果集(包含output參數(shù))的詳解
- 基于Php mysql存儲過程的詳解
- php存儲過程調(diào)用實(shí)例代碼
- PHP得到mssql的存儲過程的輸出參數(shù)功能實(shí)現(xiàn)
- Php Mssql操作簡單封裝支持存儲過程
- PHP MSSQL 存儲過程的方法
- 用PHP調(diào)用Oracle存儲過程的方法
- PHP調(diào)用MySQL的存儲過程的實(shí)現(xiàn)代碼
- php調(diào)用mysql存儲過程
- PHP調(diào)用存儲過程返回值不一致問題的解決方法分析
相關(guān)文章
關(guān)于php中的json_encode()和json_decode()函數(shù)的一些說明
下面小編就為大家?guī)硪黄P(guān)于php中的json_encode()和json_decode()函數(shù)的一些說明。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-11-11
laravel框架語言包拓展實(shí)現(xiàn)方法分析
這篇文章主要介紹了laravel框架語言包拓展實(shí)現(xiàn)方法,結(jié)合實(shí)例形式分析了laravel語言包的具體配置與使用方法,需要的朋友可以參考下2019-11-11
ThinkPHP行為擴(kuò)展Behavior應(yīng)用實(shí)例詳解
這篇文章主要介紹了ThinkPHP行為擴(kuò)展Behavior應(yīng)用實(shí)例,對于讀者深入了解ThinkPHP框架程序設(shè)計(jì)大有幫助,需要的朋友可以參考下2014-07-07
php從數(shù)據(jù)庫中讀取特定的行(實(shí)例)
下面小編就為大家?guī)硪黄猵hp從數(shù)據(jù)庫中讀取特定的行(實(shí)例)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-06-06
純php打造的tab選項(xiàng)卡效果代碼(不用js)
用php實(shí)現(xiàn)的tab選項(xiàng)卡效果,根據(jù)get判斷,獲取get生成css與對應(yīng)的內(nèi)容,當(dāng)然效率肯定沒有靜態(tài)的好,這里給出的思路與方法。2010-12-12
PHP 返回?cái)?shù)組后處理方法(開戶成功后彈窗提示)
這篇文章主要介紹了PHP 返回?cái)?shù)組后處理方法(開戶成功后彈窗提示),需要的朋友可以參考下2017-07-07

