php从内存中获取源码_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源码分析之变量的存储过程分解的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: c语言中文件是如何存储的,急求如何将下列
- 下一篇: php有哪些高级扩展,php扩展有哪些