找回密码
 立即注册

QQ登录

只需一步,快速开始

查看: 491|回复: 0
打印 上一主题 下一主题

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

[复制链接]

2487

主题

2487

帖子

7391

积分

论坛元老

Rank: 8Rank: 8

积分
7391
跳转到指定楼层
楼主
发表于 2018-2-14 05:56:35 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式

            PHP代码如下:
[U]复制代码[/U] 代码如下php_var = 1;  
对应C的代码是:
[U]复制代码[/U] 代码如下: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的结构如下:
[U]复制代码[/U] 代码如下:
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的结构如下:
[U]复制代码[/U] 代码如下:
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);//变量初始化 相关宏如下: //初始化
[U]复制代码[/U] 代码如下:
#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;  
展开后为:
[U]复制代码[/U] 代码如下:
(c_var) = (zval *) emalloc(sizeof(zval));  //分配内存  
(c_var)-> refcount = 1;  //引用计数初始化  
(c_var)-> is_ref = 0; //是否引用  
可以看到其作用就是分配内存,初始化refcount,is_ref
三.下面看第三行 ZVAL_LONG(c_var,1) 相关宏为:
[U]复制代码[/U] 代码如下:
//定义值  
#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  
展开后为:
[U]复制代码[/U] 代码如下:
(* c_var).type = IS_LONG;  
(* c_var).value = 1;  
四:接下来看第四行: ZEND_SET_SYMBOL( EG(active_symbol_table), “php_var”, c_var); 首先说明下PHP的变量是存在一个hashtable里的
[U]复制代码[/U] 代码如下:
struct _zend_executor_globals {   
        ….  
        HashTable symbol_table;//全局变量的符号表   
        HashTable *active_symbol_table;//局部变量的符号表   
        …..  
    };   
Hashtable的Key为变量的名称,即php_var,值为指向PHP变量的指针,即c_var指针; 相关宏为:
[U]复制代码[/U] 代码如下:
#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调用mysql存储过程
  • PHP调用MySQL的存储过程的实现代码
  • 用PHP调用Oracle存储过程的方法
  • PHP MSSQL 存储过程的方法
  • Php Mssql操作简单封装支持存储过程
  • PHP得到mssql的存储过程的输出参数功能实现
  • php存储过程调用实例代码
  • 基于Php mysql存储过程的详解
  • PHP调用MsSQL Server 2012存储过程获取多结果集(包含output参数)的详解
  • PHP调用MySQL存储过程并返回值的方法
  • php调用mysql存储过程实例分析
  • PHP调用存储过程返回值不一致问题的解决方法分析
            
  • 分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
    收藏收藏
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    用户反馈
    客户端