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;
  // ...
  }
}
登录后复制
PHP源码 ==(interpreter)==> opcode(PHP: Zend Engine 2 Opcodes,理解成汇编) ==(zend engine)==> 机器码计算机只能执行机器码,但是并不是把PHP代码直接转为机器码执行,而是转化为一条条的opcode,其实opcode对应的是一个C函数,执行一条opcode就是执行这条opcode对应的C函数,而这个C函数已经被编译为机器码了
09-02 03:03