PHP内核探索:如何执行PHP脚本
来源:互联网 发布:java员工工资管理系统 编辑:程序博客网 时间:2024/05/24 04:52
前面介绍了PHP的生命周期,PHP的SAPI,SAPI处于PHP整个架构较上层,而真正脚本的执行主要由Zend引擎来完成, 这一小节我们介绍PHP脚本的执行。
目前编程语言可以分为两大类:
- 第一类是像C/C++, .NET, Java之类的编译型语言, 它们的共性是:运行之前必须对源代码进行编译,然后运行编译后的目标文件。
- 第二类比如PHP, Javascript, Ruby, Python这些解释型语言, 他们都无需经过编译即可“运行”。
虽然可以理解为直接运行,但它们并不是真的直接就被能被机器理解, 机器只能理解机器语言,那这些语言是怎么被执行的呢, 一般这些语言都需要一个解释器, 由解释器来执行这些源码, 实际上这些语言还是会经过编译环节,只不过它们一般会在运行的时候实时进行编译。为了效率,并不是所有语言在每次执行的时候都会重新编译一遍, 比如PHP的各种opcode缓存扩展(如APC, xcache, eAccelerator等),比如Python会将编译的中间文件保存成pyc/pyo文件, 避免每次运行重新进行编译所带来的性能损失。
PHP的脚本的执行也需要一个解释器, 比如命令行下的php程序,或者apache的mod_php模块等等。 前面提到了PHP的SAPI接口, 下面就以PHP命令行程序为例解释PHP脚本是怎么被执行的。 例如如下的这段PHP脚本:
1
<?php
2
$str
=
"Hello, nowamagic!\n"
;
3
echo
$str
;
4
?>
假设上面的代码保存在名为hello.php的文件中, 用PHP命令行程序执行这个脚本:
1
$ php ./hello.php
这段代码的输出显然是Hello, nowamagic!, 那么在执行脚本的时候PHP/Zend都做了些什么呢? 这些语句是怎么样让php输出这段话的呢? 下面将一步一步的进行介绍。
程序的执行
- 如上例中, 传递给php程序需要执行的文件, php程序完成基本的准备工作后启动PHP及Zend引擎, 加载注册的扩展模块。
- 初始化完成后读取脚本文件,Zend引擎对脚本文件进行词法分析,语法分析。然后编译成opcode执行。 如过安装了apc之类的opcode缓存, 编译环节可能会被跳过而直接从缓存中读取opcode执行。
PHP在读取到脚本文件后首先对代码进行词法分析,PHP的词法分析器是通过lex生成的, 词法规则文件在$PHP_SRC/Zend/zend_language_scanner.l, 这一阶段lex会会将源代码按照词法规则切分一个一个的标记(token)。PHP中提供了一个函数token_get_all(), 该函数接收一个字符串参数, 返回一个按照词法规则切分好的数组。 例如将上面的php代码作为参数传递给这个函数:
1
<?php
2
$code
=<<<PHP_CODE
3
<?php
4
$str
=
"Hello, nowamagic\n"
;
5
echo
$str
;
6
PHP_CODE;
7
8
var_dump(token_get_all(
$code
));
9
?>
运行上面的脚本你将会看到一如下的输出:
01
array (
02
0 =>
03
array (
04
0 => 368,
// 脚本开始标记
05
1 => '<?php
// 匹配到的字符串
06
',
07
2 => 1,
08
),
09
1 =>
10
array (
11
0 => 371,
12
1 =>
' '
,
13
2 => 2,
14
),
15
2 =>
'='
,
16
3 =>
17
array (
18
0 => 371,
19
1 =>
' '
,
20
2 => 2,
21
),
22
4 =>
23
array (
24
0 => 315,
25
1 => '"Hello, nowamagic
26
"',
27
2 => 2,
28
),
29
5 =>
';'
,
30
6 =>
31
array (
32
0 => 371,
33
1 => '
34
',
35
2 => 3,
36
),
37
7 =>
38
array (
39
0 => 316,
40
1 =>
'echo'
,
41
2 => 4,
42
),
43
8 =>
44
array (
45
0 => 371,
46
1 =>
' '
,
47
2 => 4,
48
),
49
9 =>
';'
,
这也是Zend引擎词法分析做的事情,将代码切分为一个个的标记,然后使用语法分析器(PHP使用bison生成语法分析器, 规则见$PHP_SRC/Zend/zend_language_parser。y), bison根据规则进行相应的处理, 如果代码找不到匹配的规则,也就是语法错误时Zend引擎会停止,并输出错误信息。 比如缺少括号,或者不符合语法规则的情况都会在这个环节检查。 在匹配到相应的语法规则后,Zend引擎还会进行编译, 将代码编译为opcode, 完成后,Zend引擎会执行这些opcode, 在执行opcode的过程中还有可能会继续重复进行编译-执行, 例如执行eval,include/require等语句, 因为这些语句还会包含或者执行其他文件或者字符串中的脚本。
例如上例中的echo语句会编译为一条ZEND_ECHO指令, 执行过程中,该指令由C函数zend_print_variable(zval* z)执行,将传递进来的字符串打印出来。 为了方便理解, 本例中省去了一些细节,例如opcode指令和处理函数之间的映射关系等。 后面的章节将会详细介绍。
如果想直接查看生成的Opcode,可以使用php的vld扩展查看。扩展下载地址: http://pecl.php.net/package/vld。Win下需要自己编译生成dll文件。
有关PHP脚本编译执行的细节,请阅读后面有关词法分析,语法分析及opcode编译相关内容。
- PHP内核探索:如何执行PHP脚本
- PHP内核探索:如何执行PHP脚本
- PHP内核探索:如何执行PHP脚本
- 13.PHP内核探索:如何执行PHP脚本
- PHP内核探索:PHP脚本的执行细节
- 14.PHP内核探索:PHP脚本的执行细节
- PHP内核探索:PHP脚本的执行细节
- 13-如何执行PHP脚本
- PHP内核探索:嵌入式PHP
- PHP内核探索1
- PHP 内核探索
- PHP内核探索
- php内核探索-常量
- 17.PHP内核探索:解释器的执行过程
- php内核探索笔记-解释器的执行过程
- PHP内核探索:解释器的执行过程
- PHP程序加速探索[4]--测试--脚本执行速度测试
- PHP程序加速探索[5]--脚本执行速度测试2
- 网络请求库(ion、volley、retrofit)和图片加载库(glide、picasso、fresco)
- "-"控件通知到父窗口层处理.
- windows无法连接到group policy client服务.此问题阻止标准用户登陆系统.
- STUN和TURN技术浅析
- JSP内置对象学习案例:在线考试系统
- PHP内核探索:如何执行PHP脚本
- 云集微店开店的优势
- form 中点击button事件
- Android Listview
- Android之Glide(非常好用的图片加载框架)
- lightoj 1341__Aladdin and the Flying Carpet
- 练习(一)
- 图像有用区域
- 程序猿为什么不喜欢写文档?