当前位置:首页 > CMS教程 > Thinkphp > 列表

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

发布:smiling 来源: PHP粉丝网  添加日期:2021-03-09 14:34:51 浏览: 评论:0 

这篇文章主要介绍了PHP源码分析之变量的存储过程分解,本文针对PHP变量声明后,解释器在背后的一系列动作做了分解,需要的朋友可以参考下。

PHP代码如下:

$php_var = 1;

对应C的代码是:

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

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

  1. struct _zval_struct {  
  2.     /* Variable information */  
  3.     zvalue_value value;     /* 变量的值 */  
  4.     zend_uint refcount;     /* 引用计数,垃圾回收的时候用到 */  
  5.     zend_uchar type;        /* 变量类型 */  
  6.     zend_uchar is_ref;      /* 是否为引用变量 */  
  7. };  
  8. typedef struct _zval_struct zval;  

其中值zvalue_value的结构如下:

  1. typedef union _zvalue_value {  
  2.     long lval;              /* 长整形*/  
  3.     double dval;            /* 双精度类型 */  
  4.     struct {                  /* 字符串类型的值 */  
  5.         char *val;              
  6.         int len;  
  7.     } str;  
  8.     HashTable *ht;              /* 数组类型的值 */  
  9.     zend_object_value obj;     /*对象类型的值*/  
  10. } zvalue_value;  

二.接下来看第二行: 

  1. MAKE_STD_ZVAL(new_val);//变量初始化 相关宏如下: //初始化
  2. #define MAKE_STD_ZVAL(zv)                \  
  3.     ALLOC_ZVAL(zv); \  
  4.     INIT_PZVAL(zv);  
  5.  
  6. #define ALLOC_ZVAL(z)   \  
  7.     ZEND_FAST_ALLOC(z, zval, ZVAL_CACHE_LIST)  
  8.  
  9. #define ZEND_FAST_ALLOC(p, type, fc_type)   \  
  10.     (p) = (type *) emalloc(sizeof(type))  
  11.  
  12. #define INIT_PZVAL(z)       \  
  13.     (z)->refcount = 1;      \  
  14.     (z)->is_ref = 0;  

展开后为:

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

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

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

  1. //定义值  
  2. #define ZVAL_LONG(z, l) {           \  
  3.      Z_TYPE_P(z) = IS_LONG;      \  
  4.      Z_LVAL_P(z) = l;            \  
  5. }  
  6. #define Z_TYPE_P(zval_p)    Z_TYPE(*zval_p)  
  7. #define Z_TYPE(zval)        (zval).type  
  8. #define Z_LVAL_P(zval_p)    Z_LVAL(*zval_p)  
  9. #define Z_LVAL(zval)            (zval).value.lval  

展开后为:

  1. (* c_var).type = IS_LONG;  
  2. (* c_var).value = 1;  

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

  1. struct _zend_executor_globals {    
  2.         ….  
  3.         HashTable symbol_table;//全局变量的符号表    
  4.         HashTable *active_symbol_table;//局部变量的符号表    
  5.         …..  
  6.     }; 

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

  1. #define ZEND_SET_SYMBOL(symtable, name, var)          \   {                                                     \  
  2.         char *_name = (name);                         \  
  3.         ZEND_SET_SYMBOL_WITH_LENGTH(symtable, _name, strlen(_name)+1, var, 1, 0);   \  
  4. }  
  5. //主要的实现为下面这个函数:  
  6. #define ZEND_SET_SYMBOL_WITH_LENGTH(symtable, name, name_length, var, _refcount, _is_ref)                                                       \  
  7.     {                                                                         
  8.         zval **orig_var;                                        \   
  9.         if (zend_hash_find(symtable, (name), (name_length), (void **) &orig_var)==SUCCESS                                                         \  
  10.             && PZVAL_IS_REF(*orig_var)) {                     \  
  11.             (var)->refcount = (*orig_var)->refcount;                  \  
  12.             (var)->is_ref = 1;                                \  
  13.             if (_refcount) {                                      \  
  14.                 (var)->refcount += _refcount-1;               \  
  15.             }                                             \  
  16.             zval_dtor(*orig_var);                             \  
  17.             **orig_var = *(var);                                  \  
  18.             FREE_ZVAL(var);                               \  
  19.         } else {                                              \  
  20.             (var)->is_ref = _is_ref;                              \  
  21.             if (_refcount) {                                      \  
  22.                 (var)->refcount = _refcount;                      \  
  23.             }                                             \  
  24.             zend_hash_update(symtable, (name), (name_length), &(var), sizeof(zval *), NULL);                                                           \  
  25.         }                                                  \  
  26.     } 

该函数的功能是:

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

a. 将原来变量的引用计数refcount,is_ref信息赋给c_var;

b. 释放掉原来变量zvalue的值,比如原来其值指向的是一个mysql连接资源,则释放该资源。

c. 将c_var指向的变量赋值给原来的变量 d. 释放c_var的内存空间 这样保证了,如果变量被应用,值一起改变。比如如果前面有$b=&a;

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

Tags: PHP变量存储

分享到: