php先被Zend引擎编译为opcode,那么opcode又是如何被zend_execute执行的呢?最终是被编译为机器码了吗?具体过程是如何的?
回复内容:
既然题主提到了Zend那么这个问题是特指Zend Engine所实现的PHP,而不指任何其它实现(例如HHVM、Hippy、Quercus或JPHP)。
到PHP 7.0为止,Zend Engine的正式发行的版本从来都是通过解释器实现(*)。Zend的字节码指令一条条进入解释器,一条条字节码指令被其opcode对应的C语言写的函数所执行。就这么简单。
简易工作流程:
[ PHP源码 ]
=> 词法分析
/ 语法分析
-> [ 抽象语法树(AST)
]
=> 字节码编译器
-> [ Zend字节码(指令集为 Zend opcodes
) ]
=> 字节码解释器
-> [ 程序运行结果 ]
具体解释器是如何从opcode分派到其对应的函数,Zend Engine可以配置使用几种不同的方式:call threading、switch threading、computed goto threading。这些名字分别是什么意思,请参考这篇文章:Threaded Code
以最简单的switch-threading举例,这种解释器的基本形式就是这样的:
while (TRUE) { int opcode = *program_counter; switch (opcode) { case ZEND_ADD: // execute add ... program_counter++; // next opcode break; case ZEND_SUB: // execute sub ... program_counter++; // next opcode break; // ... } }
Copy after login
PHP源码 ==(interpreter)==> opcode(PHP: Zend Engine 2 Opcodes
,理解成汇编) ==(zend engine)==> 机器码 计算机只能执行机器码,但是并不是把PHP代码直接转为机器码执行,而是转化为一条条的opcode,其实opcode对应的是一个C函数,执行一条opcode就是执行这条opcode对应的C函数,而这个C函数已经被编译为机器码了