<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>潘锦的空间 &#187; PHP脚本的执行</title>
	<atom:link href="https://www.phppan.com/tag/php%e8%84%9a%e6%9c%ac%e7%9a%84%e6%89%a7%e8%a1%8c/feed/" rel="self" type="application/rss+xml" />
	<link>https://www.phppan.com</link>
	<description>SaaS SaaS架构 团队管理 技术管理 技术架构 PHP 内核 扩展 项目管理</description>
	<lastBuildDate>Sat, 25 Apr 2026 00:56:17 +0000</lastBuildDate>
	<language>zh-CN</language>
		<sy:updatePeriod>hourly</sy:updatePeriod>
		<sy:updateFrequency>1</sy:updateFrequency>
	<generator>https://wordpress.org/?v=3.9.40</generator>
	<item>
		<title>TIPI0203-PHP脚本的执行</title>
		<link>https://www.phppan.com/2011/01/tipi0203/</link>
		<comments>https://www.phppan.com/2011/01/tipi0203/#comments</comments>
		<pubDate>Fri, 28 Jan 2011 06:24:27 +0000</pubDate>
		<dc:creator><![CDATA[admin]]></dc:creator>
				<category><![CDATA[PHP]]></category>
		<category><![CDATA[PHP脚本的执行]]></category>
		<category><![CDATA[TIPI]]></category>
		<category><![CDATA[深入理解PHP内核]]></category>

		<guid isPermaLink="false">http://www.phppan.com/?p=1256</guid>
		<description><![CDATA[在前面的章节介绍了PHP的生命周期，PHP的SAPI，这些内容都是处于上层的，在这个下面是对于PHP本身的解析 [&#8230;]]]></description>
				<content:encoded><![CDATA[<p style="text-indent: 2em;">在前面的章节介绍了PHP的生命周期，PHP的SAPI，这些内容都是处于上层的，在这个下面是对于PHP本身的解析和执行。这一小节我们介绍PHP脚本的执行。</p>
<p style="text-indent: 2em;">目前的编程语言可以分为两大类:</p>
<ul>
<li>第一类是像C/C++, .NET, Java之类的编译型语言, 它们的共性是: 运行之前必须对源代码进行编译,然后运行编译后的目标文件.</li>
<li>第二类比如:PHP, Javascript, Ruby, Python这些解释型语言, 他们都无需经过编译即可&#8221;运行&#8221;. 虽然可以理解为直接运行, 但它们并不是真的直接就被能被机器理解, 机器只能理解机器语言,那这些语言是怎么被执行的呢, 一般这些语言都需要一个<strong>解释器</strong>, 由解释器来执行这些源码, 实际上这些语言还是会经过编译环节, 只不过它们一般会在运行的时候实时进行编译. 为了效率,并不是所有语言在每次执行的时候都会重新编译一遍, 比如PHP的各种opcode缓存扩展(如APC, xcache, eAccelerator等),比如Python会将编译的中间文件保存成pyc/pyo文件,避免每次运行重新进行编译所带来的性能损失.</li>
</ul>
<p style="text-indent: 2em;">PHP的脚本的执行也需要一个解释器, 比如命令行下的php程序,或者apache的mod_php模块等等. 前一节提到了PHP的SAPI接口, 下面就以PHP命令行程序为例解释PHP脚本是怎么被执行的. 例如如下的这段PHP脚本:</p>
<pre style="background-color: #333333; color: #ffffff; font: normal normal normal 13px/normal 'Bitstream Vera Sans Mono', 'DejaVu Sans Mono', 'Courier New', monospace; overflow-x: auto; overflow-y: auto; padding: 10px;"><span style="color: #cc7833;">&lt;?php</span>
<span style="color: #6d9cbe;">$str</span> <span style="color: #e0882f;">=</span> <span style="color: #99ff00;">"Hello, Tipi!<span>\n</span>"</span><span style="color: #e0882f;">;</span>
<a style="color: #1299da; text-decoration: none;" href="http://www.php.net/echo"><span style="color: #e2392d;">echo</span></a> <span style="color: #6d9cbe;">$str</span><span style="color: #e0882f;">;</span></pre>
<p style="text-indent: 2em;">假设上面的代码保存在名为hello.php的文件中, 用PHP命令行程序执行这个脚本:</p>
<pre style="background-color: #333333; color: #ffffff; font: normal normal normal 13px/normal 'Bitstream Vera Sans Mono', 'DejaVu Sans Mono', 'Courier New', monospace; overflow-x: auto; overflow-y: auto; padding: 10px;">$ php <span>--help</span>  <span style="color: #bd48b3;"># 显示php程序可以接受的参数</span>
$ php .<span style="color: #e0882f;">/</span>hello.php</pre>
<p style="text-indent: 2em;">这段代码的输出显然是Hello, Tipi!, 那么在执行脚本的时候PHP/Zend都做了些什么呢? 这些语句是怎么样让php输出这段话的呢? 下面将一步一步的进行介绍.</p>
<h2 style="font-weight: bold; font-family: 'Microsoft YaHei', Helvetica, Arial, sans-serif; font-size: 1.2em; color: #333333;">程序的执行</h2>
<ol>
<li>如上例中, 传递给php程序需要执行的文件, php程序完成基本的准备工作后启动PHP及Zend引擎, 加载注册的扩展模块.</li>
<li>初始化完成后读取脚本文件,Zend引擎对脚本文件进行词法分析,语法分析. 然后编译成opcode 执行. 如果安装了apc之类的opcode缓存, 编译环节可能会被跳过而直接从缓存中读取opcode执行.</li>
</ol>
<h3 style="font-weight: bold; font-family: 'Microsoft YaHei', Helvetica, Arial, sans-serif; font-size: 1.1em; color: #333333;">脚本的编译执行</h3>
<p style="text-indent: 2em;">PHP在读取到脚本文件后首先对代码进行词法分析, PHP的词法分析器是通过lex生成的, 词法规则文件在$PHP_SRC/Zend/zend_language_scanner.l, 这一阶段lex会会将源代码按照词法规则切分一个一个的标记(token). PHP中提供了一个函数token_get_all(), 该函数接收一个字符串参数, 返回一个按照词法规则切分好的数组. 例如将上面的php代码作为参数传递给这个函数:</p>
<pre style="background-color: #333333; color: #ffffff; font: normal normal normal 13px/normal 'Bitstream Vera Sans Mono', 'DejaVu Sans Mono', 'Courier New', monospace; overflow-x: auto; overflow-y: auto; padding: 10px;"><span style="color: #cc7833;">&lt;?php</span>
<span style="color: #6d9cbe;">$code</span> <span style="color: #e0882f;">=</span><span>&lt;&lt;&lt;PHP_CODE
&lt;?php
$str = "Hello, Tipi\n";
echo $str;
PHP</span>_CODE<span style="color: #e0882f;">;</span>

<a style="color: #1299da; text-decoration: none;" href="http://www.php.net/var_dump"><span style="color: #e2392d;">var_dump</span></a><span style="color: #ffffff;">(</span><a style="color: #1299da; text-decoration: none;" href="http://www.php.net/token_get_all"><span style="color: #e2392d;">token_get_all</span></a><span style="color: #ffffff;">(</span><span style="color: #6d9cbe;">$code</span><span style="color: #ffffff;">)</span><span style="color: #ffffff;">)</span><span style="color: #e0882f;">;</span></pre>
<p style="text-indent: 2em;">运行上面的脚本你将会看到一如下的输出</p>
<pre style="background-color: #333333; color: #ffffff; font: normal normal normal 13px/normal 'Bitstream Vera Sans Mono', 'DejaVu Sans Mono', 'Courier New', monospace; overflow-x: auto; overflow-y: auto; padding: 10px;">array (
  0 =&gt;
  array (
    0 =&gt; 368,       // 脚本开始标记
    1 =&gt; '&lt;?php     // 匹配到的字符串
',
    2 =&gt; 1,
  ),
  1 =&gt;
  array (
    0 =&gt; 371,
    1 =&gt; ' ',
    2 =&gt; 2,
  ),
  2 =&gt; '=',
  3 =&gt;
  array (
    0 =&gt; 371,
    1 =&gt; ' ',
    2 =&gt; 2,
  ),
  4 =&gt;
  array (
    0 =&gt; 315,
    1 =&gt; '"Hello, Tipi
"',
    2 =&gt; 2,
  ),
  5 =&gt; ';',
  6 =&gt;
  array (
    0 =&gt; 371,
    1 =&gt; '
',
    2 =&gt; 3,
  ),
  7 =&gt;
  array (
    0 =&gt; 316,
    1 =&gt; 'echo',
    2 =&gt; 4,
  ),
  8 =&gt;
  array (
    0 =&gt; 371,
    1 =&gt; ' ',
    2 =&gt; 4,
  ),
  9 =&gt; ';',</pre>
<p style="text-indent: 2em;">这也是Zend引擎词法分析做的事情,将代码切分为一个个的标记,然后使用语法分析器(PHP使用yacc生成语法分析器, 规则见$PHP_SRC/Zend/zend_language_parser.y), yacc根据规则进行相应的处理, 如果代码找不到匹配的规则,也就是语法错误时Zend引擎会停止,并输出错误信息. 比如缺少括号,或者不符合语法规则的情况都会在这个环节检查. 在匹配到相应的语法规则后,Zend引擎还会进行编译, 将代码编译为opcode, 完成后,Zend引擎会执行这些opcode, 在执行opcode的过程中还有可能会继续重复进行编译-执行, 例如执行eval,include/require等语句, 因为这些语句还会包含或者执行其他文件或者字符串中的脚本.</p>
<p style="text-indent: 2em;">例如上例中的echo语句会编译为一条ZEND_ECHO指令, 执行过程中,该指令由C函数zend_print_variable(zval* z)执行,将传递进来的字符串打印出来. 为了方便理解, 本例中省去了一些细节,例如opcode指令和处理函数之间的映射关系等. 后面的章节将会详细介绍.</p>
<p style="text-indent: 2em;">如果想直接查看生成的Opcode，可以使用php的vld扩展查看。扩展下载地址: <a style="color: #1299da; text-decoration: underline;" href="http://pecl.php.net/package/vld">http://pecl.php.net/package/vld</a>。Win下需要自己编译生成dll文件。</p>
<p style="text-indent: 2em;">作者：TIPI团队</p>
]]></content:encoded>
			<wfw:commentRss>https://www.phppan.com/2011/01/tipi0203/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>
