• 欢迎访问搞代码网站,推荐使用最新版火狐浏览器和Chrome浏览器访问本网站!
  • 如果您觉得本站非常有看点,那么赶紧使用Ctrl+D 收藏搞代码吧

PHP源码分析之变量的存储过程分解_php实例

php 搞代码 3年前 (2022-01-26) 44次浏览 已收录 0个评论

PHP代码如下:

$php_var = 1;  

对应C的代码是:

zval* c_var;    //定义PHP变量指针  <br />MAKE_STD_ZVAL(c_var);  //初始化PHP变量  <br />ZVAL_LONG(c_var,1) ;//赋值  <br />ZEND_SET_SYMBL( EG(active_symbol_table), " php_var ", c_var);//注册到全局变量符号表

一.首先看第一行: zval* c_var;//申明一个zval指针c_var; zval的结构如下:

<br />struct _zval_struct {  <br />    /* Variable information */  <br />    zvalue_value value;     /* 变量的值 */  <br />    zend_uint refcount;     /* 引用计数,垃圾回收的时候用到 */  <br />    zend_uchar type;        /* 变量类型 */  <br />    zend_uchar is_ref;      /* 是否为引用变量 */  <br />};  <br />typedef struct _zval_struct zval;  <br />

其中值zvalue_value的结构如下:

<br />typedef union _zvalue_value {  <br />    long lval;              /* 长整形*/  <br />    double dval;            /* 双精度类型 */  <br />    struct {                  /* 字符串类型的值 */  <br />        char *val;              <br />        int len;  <br />    } str;  <br />    HashTable *ht;              /* 数组类型的值 */  <br />    zend_object_value obj;     /*对象类型的值*/  <br />} zvalue_value;  <br />

二.接下来看第二行: MAKE_STD_ZVAL(new_val);//变量初始化 相关宏如下: //初始化

<br />#define MAKE_STD_ZVAL(zv)                \  <br />    ALLOC_ZVAL(zv); \  <br /><mark style="color:transparent">本文来源gaodaimacom搞#^代%!码&网*</mark><pre>搞代gaodaima码

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;

展开后为:

<br />(c_var) = (zval *) emalloc(sizeof(zval));  //分配内存  <br />(c_var)-> refcount = 1;  //引用计数初始化  <br />(c_var)-> is_ref = 0; //是否引用  <br />

可以看到其作用就是分配内存,初始化refcount,is_ref

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

<br />//定义值  <br />#define ZVAL_LONG(z, l) {           \  <br />     Z_TYPE_P(z) = IS_LONG;      \  <br />     Z_LVAL_P(z) = l;            \  <br />}  <br />#define Z_TYPE_P(zval_p)    Z_TYPE(*zval_p)  <br />#define Z_TYPE(zval)        (zval).type  <br />#define Z_LVAL_P(zval_p)    Z_LVAL(*zval_p)  <br />#define Z_LVAL(zval)            (zval).value.lval  <br />

展开后为:

<br />(* c_var).type = IS_LONG;  <br />(* c_var).value = 1;  <br />

四:接下来看第四行: ZEND_SET_SYMBOL( EG(active_symbol_table), “php_var”, c_var); 首先说明下PHP的变量是存在一个hashtable里的

<br />struct _zend_executor_globals {    <br />        ….  <br />        HashTable symbol_table;//全局变量的符号表    <br />        HashTable *active_symbol_table;//局部变量的符号表    <br />        …..  <br />    };    <br />

Hashtable的Key为变量的名称,即php_var,值为指向PHP变量的指针,即c_var指针; 相关宏为:

<br />#define ZEND_SET_SYMBOL(symtable, name, var)          \   {                                                     \  <br />        char *_name = (name);                         \  <br />        ZEND_SET_SYMBOL_WITH_LENGTH(symtable, _name, strlen(_name)+1, var, 1, 0);   \  <br />}  <br />//主要的实现为下面这个函数:  <br />#define ZEND_SET_SYMBOL_WITH_LENGTH(symtable, name, name_length, var, _refcount, _is_ref)                                                       \  <br />    {                                                                         <br />        zval **orig_var;                                        \   <br />        if (zend_hash_find(symtable, (name), (name_length), (void **) &orig_var)==SUCCESS                                                         \  <br />            && PZVAL_IS_REF(*orig_var)) {                     \  <br />            (var)->refcount = (*orig_var)->refcount;                  \  <br />            (var)->is_ref = 1;                                \  <br />            if (_refcount) {                                      \  <br />                (var)->refcount += _refcount-1;               \  <br />            }                                             \  <br />            zval_dtor(*orig_var);                             \  <br />            **orig_var = *(var);                                  \  <br />            FREE_ZVAL(var);                               \  <br />        } else {                                              \  <br />            (var)->is_ref = _is_ref;                              \  <br />            if (_refcount) {                                      \  <br />                (var)->refcount = _refcount;                      \  <br />            }                                             \  <br />            zend_hash_update(symtable, (name), (name_length), &(var), sizeof(zval *), NULL);                                                           \  <br />        }                                                  \  <br />    }             <br />

该函数的功能是:
1. 如果全局符号表已经存在该变量且是引用类型,则

a. 将原来变量的引用计数refcount,is_ref信息赋给c_var;
b. 释放掉原来变量zvalue的值,比如原来其值指向的是一个mysql连接资源,则释放该资源。
c. 将c_var指向的变量赋值给原来的变量 d. 释放c_var的内存空间 这样保证了,如果变量被应用,值一起改变。比如如果前面有$b=&a;

2. 如果全局符号表不存在该变量或者存在该变量但不是引用变量,则直接改变其值。


搞代码网(gaodaima.com)提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发送到邮箱[email protected],我们会在看到邮件的第一时间内为您处理,或直接联系QQ:872152909。本网站采用BY-NC-SA协议进行授权
转载请注明原文链接:PHP源码分析之变量的存储过程分解_php实例

喜欢 (0)
[搞代码]
分享 (0)
发表我的评论
取消评论

表情 贴图 加粗 删除线 居中 斜体 签到

Hi,您需要填写昵称和邮箱!

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址