首頁  >  文章  >  後端開發  >  PHP7核心剖析5之 PHP程式碼的編譯

PHP7核心剖析5之 PHP程式碼的編譯

不言
不言原創
2018-04-13 15:02:201666瀏覽

這篇文章的內容介紹的是關於PHP7核心剖析5之PHP程式碼的編譯 ,現在分享給大家,有需要的朋友可以參考一下

1.PHP程式碼的編譯

PHP的解析過程任務就是將PHP程式碼轉換成opcode數組,程式碼裡的所有資訊都保存在opcode數組中,然後將opcode數組交給zend引擎執行,opcode就是核心具體執行的指令,例如賦值、加減運算、函數呼叫等,每一條opcode都對應一個處理handle,這些handler是事先定義好的C函數。

PHP7核心剖析5之 PHP程式碼的編譯


#2.PHP程式碼->抽象語法樹(AST)

PHP使用re2c、bison完成这个阶段的工作:
    re2c: 词法分析器,将输入分割为一个个有意义的词块,称为token
    bison: 语法分析器,确定词法分析器分割出的token是如何彼此关联的

PHP7核心剖析5之 PHP程式碼的編譯

词法、语法解析过程

1.yyparse(zendparse)调用yylex(zendlex),当读取到一个合法的token时,返回值为token类型
2.yylex调用lex_scan读取合法的token值
3.yyparse将token类型与token值构造抽象语法树,最后将根节点保存到CG(compiler_globals ,Zend编译器相关的全局变量)的ast中


#3.AST節點結構

typedef struct _zend_ast   zend_ast;

//普通节点类型
struct _zend_ast {
    zend_ast_kind kind;  //节点类型
    zend_ast_attr attr;  //节点附加属性
    uint32_t lineno;    //行号
    zend_ast *child[1];  //子节点数组
};

//普通节点类型,但有子节点的个数
typedef struct _zend_ast_list {
    zend_ast_kind kind; //节点类型
    zend_ast_attr attr; //节点附加属性
    uint32_t lineno; //行号
    uint32_t children; //子节点数量
    zend_ast *child[1];//子节点数组
} zend_ast_list;

//函数、类的ast节点结构
typedef struct _zend_ast_decl {
    zend_ast_kind kind; //节点类型
    zend_ast_attr attr; //节点附加属性
    uint32_t start_lineno; //开始行号
    uint32_t end_lineno;   //结束行号
    uint32_t flags;
    unsigned char *lex_pos;
    zend_string *doc_comment;
    zend_string *name;
    zend_ast *child[4]; //类中会将继承的父类、实现的接口以及类中的语句解析保存在child中
} zend_ast_decl;
实例:
$a = 123;
$b = "hi~";

echo $a,$b;

PHP7核心剖析5之 PHP程式碼的編譯


#4.zend_op_array

PHP7核心剖析5之 PHP程式碼的編譯

zend_op *opcodes; //opcode指令数组
zval *literals; //字面量(常量)数组,这些都是在PHP代码定义的一些值
zend_string **vars; //PHP变量名数组,根据变量编号可以获取相应的变量
//opcode指令结构
struct _zend_op {
    const void *handler; //指令执行handler
    znode_op op1;   //操作数1
    znode_op op2;   //操作数2
    znode_op result; //返回值
    uint32_t extended_value; 
    uint32_t lineno; 
    zend_uchar opcode;  //opcode指令
    zend_uchar op1_type; //操作数1类型
    zend_uchar op2_type; //操作数2类型
    zend_uchar result_type; //返回值类型
};


5.handler處理函數

handler為每條opcode對應的C語言所寫的處理過程,所有opcode對應的處理過程定義在zend_vm_def.h中,opcode的處理過程有三種不同的提供形式:CALL、SWITCH、GOTO,預設方式為CALL
CALL:把每种opcode负责的工作封装成不同的function,然后执行器循环调用执行
SWITCH:把所有的处理方式写到一个switch下,然后通过case不同的opcode执行具体的操作
GOTO:把所有opcode的处理方式通过C语言里面的label标签区分开,然后执行器执行的时候goto到相应的位置处理


6.抽象語法樹->Opcodes

void zend_compile_top_stmt(zend_ast *ast){
    ....
    if (ast->kind == ZEND_AST_STMT_LIST) { //第一次进来一定是这种类型
        zend_ast_list *list = zend_ast_get_list(ast);
        uint32_t i;
        for (i = 0; i < list->children; ++i) {
            zend_compile_top_stmt(list->child[i]);//list各child语句相互独立,递归编译
        }
        return;
    }
    //各语句编译入口
    zend_compile_stmt(ast);
    ....
}

1.zend_compile_top_stmt接收语法树,首先判断节点类型是否为ZEND_AST_STMT_LIST(表示当前节点下
有多个独立的节点),若是则进行递归
2.当递归结束后,调用zend_compile_stmt进行编译成opcodes

相關建議:

PHP7核心剖析4之局部變數,全域變數,常數 

#PHP7核心剖析3之變項

##PHP7核心剖析2之I/O模式

                    

以上是PHP7核心剖析5之 PHP程式碼的編譯的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn