这篇文章主要介绍了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指针; 相关宏为:
复制代码 代码如下: