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

 更新时间:2014年07月03日 08:50:53   投稿:junjie  
这篇文章主要介绍了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. 如果全局符号表不存在该变量或者存在该变量但不是引用变量,则直接改变其值。

相关文章

  • Linux下手动编译安装PHP扩展的例子分享

    Linux下手动编译安装PHP扩展的例子分享

    这篇文章主要介绍了Linux下手动编译安装PHP扩展的例子分享,本文以PDO_MYSQL为例,讲解手动编译安装PHP扩展的方法,需要的朋友可以参考下
    2014-07-07
  • 使用php批量删除数据库下所有前缀为prefix_的表

    使用php批量删除数据库下所有前缀为prefix_的表

    这篇文章主要介绍了php如何批量删除数据库下所有前缀为prefix_的表,需要的朋友可以参考下
    2014-06-06
  • php执行多个存储过程的方法【基于thinkPHP】

    php执行多个存储过程的方法【基于thinkPHP】

    这篇文章主要介绍了php执行多个存储过程的方法,结合实例形式分析了基于thinkPHP框架调用多个存储过程的相关操作技巧,需要的朋友可以参考下
    2016-11-11
  • CI框架常用方法小结

    CI框架常用方法小结

    这篇文章主要介绍了CI框架常用方法,总结分析了CI框架加载文件、模板、调用函数、对象等技巧,非常简单实用,需要的朋友可以参考下
    2016-05-05
  • 浅谈PHP的数据库接口和技术

    浅谈PHP的数据库接口和技术

    下面小编就为大家带来一篇浅谈PHP的数据库接口和技术。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-12-12
  • PHP微信开发之根据用户回复关键词\位置返回附近信息

    PHP微信开发之根据用户回复关键词\位置返回附近信息

    这篇文章主要为大家详细介绍了PHP微信开发之简单实现根据用户回复关键词\位置返回附近信息 ,感兴趣的小伙伴们可以参考一下
    2016-06-06
  • php通过会话控制实现身份验证实例

    php通过会话控制实现身份验证实例

    本文介绍了php通过会话控制实现身份验证实例,身份的验证是通过session提交数据的。有需要的朋友可以来了解一下。
    2016-10-10
  • PHPCMS V9 添加二级导航的思路详解

    PHPCMS V9 添加二级导航的思路详解

    这篇文章主要介绍了PHPCMS V9 添加二级导航所遇到些问题,查询导航栏的信息时返回的$r[arrchildid]与自己想象的不符,文档上说是返回子栏目id但是却有些不同。下面小编通过分享本文给大家解答下
    2016-10-10
  • CodeIgniter生成静态页的方法

    CodeIgniter生成静态页的方法

    这篇文章主要介绍了CodeIgniter生成静态页的方法,涉及CodeIgniter框架使用文件操作类生成静态文件的实现技巧,需要的朋友可以参考下
    2016-05-05
  • 大家在抢红包,程序员在研究红包算法

    大家在抢红包,程序员在研究红包算法

    微信红包在春节的火爆程度不言而喻,广告主投入5亿现金红包,与央视羊年春晚独家合作起到了巨大的推动作用。这就像一针大补丸,在短时间内给微信带来了极大的关注度与流量。下面通过本篇文章学习下抢红包算法是怎样的,小伙伴们快来一起学习吧
    2015-08-08

最新评论