Home >php教程 >php手册 >原:PHP内核函数研究之 global

原:PHP内核函数研究之 global

WBOY
WBOYOriginal
2016-06-06 19:57:071109browse

声明:本文为斯人原创,全部为作者一一分析得之,有不对的地方望赐教。 http://imsiren.com/archives/601 好久没有写博客了…最近事挺多, 换了工作,又搬了家.. 今天就来说说 我们经常用到的global语句吧.. 我们都知道,在函数体内声明的变量,作用域只是当前函

声明:本文为斯人原创,全部为作者一一分析得之,有不对的地方望赐教。

http://imsiren.com/archives/601


好久没有写博客了…最近事挺多,
换了工作,又搬了家..
今天就来说说 我们经常用到的global语句吧..
我们都知道,在函数体内声明的变量,作用域只是当前函数体中,一般情况下是访问不到外部声明的变量的.
因为全局变量和局部变量存放在不同的hashTable,
全局变量在symbol_table中,而局部变量则存在active_symbol_table中.
这样就将它们分开了,那 global是怎么实现的呢?
经过简单分析,通过RE2C&&YACC定位到 global的token

T_GLOBAL global_var_list ';'
................
global_var_list:
                global_var_list ',' global_var  { zend_do_fetch_global_variable(&$3, NULL, ZEND_FETCH_GLOBAL_LOCK TSRMLS_CC); }
        |       global_var                                              { zend_do_fetch_global_variable(&$1, NULL, ZEND_FETCH_GLOBAL_LOCK TSRMLS_CC); }
;

重点是在zend_do_fetch_global_variable这个函数中,我们进去看看,
void zend_do_fetch_global_variable(znode * varname,const znode * static_assignment,int
    fetch_type TSRMLS_DC) /* {{{ */ {
    zend_op * opline;
    znode lval;
    znode result;
    if(varname->op_type == IS_CONST) {
        if(Z_TYPE(varname->u . constant) != IS_STRING) {
            convert_to_string(&varname->u . constant);
        }
    }
    opline = get_next_op(CG(active_op_array)TSRMLS_CC);
    opline->opcode = ZEND_FETCH_W;
     /* the default mode must be Write, since fetch_simple_variable() is used to define function arguments */
    opline->result . op_type = IS_VAR;
    opline->result . u . EA . type = 0;
    opline->result . u . var = get_temporary_variable(CG(active_op_array));
    opline->op1 = * varname;
    SET_UNUSED(opline->op2);
    opline->op2 . u . EA . type = fetch_type;
    result = opline->result;
    if(varname->op_type == IS_CONST) {
        zval_copy_ctor(&varname->u . constant);
    }
    fetch_simple_variable(&lval,varname,0TSRMLS_CC);
     /* Relies on the fact that the default fetch is BP_VAR_W */
    zend_do_assign_ref(NULL,&lval,&result TSRMLS_CC);
    CG(active_op_array)->opcodes[CG(active_op_array)->last - 1] . result . u . EA . type |=
        EXT_TYPE_UNUSED;
}

1:此函数先验证我们获取的变量是否是常量,如果是常量并且不是字符串类型,则就将它转换成string类型.
2:此函数生成一个Op , ZEND_FETCH_W
3:释放CONST
4:fetch_simple_variable会检测是不是auto_global
最重要的地方是 ZEND_FETCH_W
通过计算 可以知道 op的函数是ZEND_FETCH_W_SPEC_CV_HANDLER


static int ZEND_FASTCALL  ZEND_FETCH_W_SPEC_CV_HANDLER(ZEND_OPCODE_HANDLER_ARGS)
{
        return zend_fetch_var_address_helper_SPEC_CV(BP_VAR_W, ZEND_OPCODE_HANDLER_ARGS_PASSTHRU);
}

zend_fetch_var_address_helper_SPEC_CV的工作:
1:转换成IS_STRING类型
2:如果是类的静态成员,则通过::获取过来,否则,通过zend_get_target_symbol_table函数返回全局变量symbol_table .
return &EG(symbol_table);
3:通过zend_hash_find 在 target_symbol_table 中查找,target_symbol_table是一个临时HashTable.用来存放 symbol_table的数据.
4: 如果没有找到,则会以写的方式 创建一条数据到target_symbol_table中.这就是,我们为什么通过global一个不存在的变量的时候不出错,并且会创建该变量的原因.

http://imsiren.com/archives/601
Statement:
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn