PHP运行过程 简单点的 浏览器 -> web服务器 -> php解释器 -> mysql服务器 -> php解释器 -> web服务器 -> 浏览器 大家握几个手 复杂点的 浏览器 -----------------------------> web服务器-> php解释器 -> mysql服务器 …. PHP运行机制 2009-10-20 16:49 先看看下面这个过程: 1. 我们从未手动开启过PHP的相关进程,它是随着Apache的启动而运行的; 2. PHP通过mod_php5.so模块和Apache相连(具体说来是SAPI,即服务器应用程序编程接口); 3. PHP总共有三个模块:内核、Zend引擎、以及扩展层; 4. PHP内核用来处理请求、文件流、错误处理等相关操作; 5. Zend引擎(ZE)用以将源文件转换成机器语言,然后在虚拟机上运行它; 6. 扩展层是一组函数、类库和流,PHP使用它们来执行一些特定的操作。比如,我们需要mysql扩展来连接MySQL数据库; 7. 当ZE执行程序时可能会需要连接若干扩展,这时ZE将控制权交给扩展,等处理完特定任务后再返还; 8. 最后,ZE将程序运行结果返回给PHP内核,它再将结果传送给SAPI层,最终输出到浏览器上。 深入探讨 等等,没有这么简单。以上过程只是个简略版,让我们再深入挖掘一下,看看幕后还发生了些什么。 Apache启动后,PHP解释程序也随之启动; PHP的启动过程有两步; 第一步是初始化一些环境变量,这将在整个SAPI生命周期中发生作用; 第二步是生成只针对当前请求的一些变量设置。 PHP启动第一步 不清楚什么第一第二步是什么?别担心,我们接下来详细讨论一下。 让我们先看看第一步,也是最主要的一步。要记住的是,第一步的操作在任何请求到达之前就发生了。 启动Apache后,PHP解释程序也随之启动;PHP调用各个扩展的MINIT方法,从而使这些扩展切换到可用状态。看看php.ini文件里打开了哪些扩展吧; MINIT的意思是“模块初始化”。各个模块都定义了一组函数、类库等用以处理其他请求。 一个典型的MINIT方法如下: PHP_MINIT_FUNCTION(extension_name){ /* Initialize functions, classes etc */ } PHP启动第二步 当一个页面请求发生时,SAPI层将控制权交给PHP层。于是PHP设置了用于回复本次请求所需的环境变量。同时,它还建立一个变量表,用来存放执行过程中产生的变量名和值。 PHP调用各个模块的RINIT方法,即“请求初始化”。一个经典的例子是Session模块的RINIT,如果在php.ini中启用了Session模块,那在调用该模块的RINIT时就会初始化$_SESSION变量,并将相关内容读入; RINIT方法可以看作是一个准备过程,在程序执行之间就会自动启动。 一个典型的RINIT方法如下: PHP_RINIT_FUNCTION(extension_name) { /* Initialize session variables, pre-populate variables, redefine global variables etc */ } PHP关闭第一步 如同PHP启动一样,PHP的关闭也分两步: 一旦页面执行完毕(无论是执行到了文件末尾还是用exit或die函数中止),PHP就会启动清理程序。它会按顺序调用各个模块的RSHUTDOWN方法。 RSHUTDOWN用以清除程序运行时产生的符号表,也就是对每个变量调用unset函数。 一个典型的RSHUTDOWN方法如下: PHP_RSHUTDOWN_FUNCTION(extension_name) { /* Do memory management, unset all variables used in the last PHP call etc */ } PHP关闭第二步 最后,所有的请求都已处理完毕,SAPI也准备关闭了,PHP开始执行第二步: PHP调用每个扩展的MSHUTDOWN方法,这是各个模块最后一次释放内存的机会。 一个典型的RSHUTDOWN方法如下: PHP_MSHUTDOWN_FUNCTION(extension_name) { /* Free handlers and persistent memory etc */ } 这样,整个PHP生命周期就结束了。要注意的是,只有在服务器没有请求的情况下才会执行“启动第一步”和“关闭第二步”。 php的生命周期 2009-10-20 16:49 1.启动apache后,php解释程序也随之启动(为什么?标注1),这时启动php将会调用各种扩展,使用MINIT方法,使这些扩展成为可用状态(如何调用?标注2),此处可通过php.ini查看启用了那些扩展 初始化的方法是 PHP_MINIT_FUNCTION(),可查找源代码得到该处的解释 标注1:apache启动读取httpd.conf 中的LoadModule php5_module modules/mod_php5.so或者LoadModule php5_module d:/php/php5apache2.dll随之便启动php解释程序 标注2:以下是一个date扩展 PHP_MINIT_FUNCTION(date) { REGISTER_INI_ENTRIES();//php.h定义宏 作用是将初始化配置引入php date_register_classes(TSRMLS_C);// /* * RFC4287, Section 3.3: http://www.ietf.org/rfc/rfc4287.txt * A Date construct is an element whose content MUST conform to the * "date-time" production in [RFC3339]. In addition, an uppercase "T" * character MUST be used to separate date and time, and an uppercase * "Z" character MUST be present in the absence of a numeric time zone offset. */ REGISTER_STRING_CONSTANT("DATE_ATOM", DATE_FORMAT_RFC3339, CONST_CS | CONST_PERSISTENT); /* * Preliminary specification: http://wp.netscape.com/newsref/std/cookie_spec.html * "This is based on RFC 822, RFC 850, RFC 1036, and RFC 1123, * with the variations that the only legal time zone is GMT * and the separators between the elements of the date must be dashes." */ REGISTER_STRING_CONSTANT("DATE_COOKIE", DATE_FORMAT_RFC850, CONST_CS | CONST_PERSISTENT);//以下都是注册字符串常量,比如时间格式 DATE_ISO8601... REGISTER_STRING_CONSTANT("DATE_ISO8601", DATE_FORMAT_ISO8601, CONST_CS | CONST_PERSISTENT); REGISTER_STRING_CONSTANT("DATE_RFC822", DATE_FORMAT_RFC822, CONST_CS | CONST_PERSISTENT); REGISTER_STRING_CONSTANT("DATE_RFC850", DATE_FORMAT_RFC850, CONST_CS | CONST_PERSISTENT); REGISTER_STRING_CONSTANT("DATE_RFC1036", DATE_FORMAT_RFC1036, CONST_CS | CONST_PERSISTENT); REGISTER_STRING_CONSTANT("DATE_RFC1123", DATE_FORMAT_RFC1123, CONST_CS | CONST_PERSISTENT); REGISTER_STRING_CONSTANT("DATE_RFC2822", DATE_FORMAT_RFC2822, CONST_CS | CONST_PERSISTENT); REGISTER_STRING_CONSTANT("DATE_RFC3339", DATE_FORMAT_RFC3339, CONST_CS | CONST_PERSISTENT); /* * RSS 2.0 Specification: http://blogs.law.harvard.edu/tech/rss * "All date-times in RSS conform to the Date and Time Specification of RFC 822, * with the exception that the year may be expressed with two characters or four characters (four preferred)" */ REGISTER_STRING_CONSTANT("DATE_RSS", DATE_FORMAT_RFC1123, CONST_CS | CONST_PERSISTENT); REGISTER_STRING_CONSTANT("DATE_W3C", DATE_FORMAT_RFC3339, CONST_CS | CONST_PERSISTENT); REGISTER_LONG_CONSTANT("SUNFUNCS_RET_TIMESTAMP", SUNFUNCS_RET_TIMESTAMP, CONST_CS | CONST_PERSISTENT); REGISTER_LONG_CONSTANT("SUNFUNCS_RET_STRING", SUNFUNCS_RET_STRING, CONST_CS | CONST_PERSISTENT); REGISTER_LONG_CONSTANT("SUNFUNCS_RET_DOUBLE", SUNFUNCS_RET_DOUBLE, CONST_CS | CONST_PERSISTENT); php_date_global_timezone_db = NULL; php_date_global_timezone_db_enabled = 0; return SUCCESS; } 2.当一个页面发生请求后,SAPI层将控制权交给php层(如何?),此时php层设置所有的环境变量(怎样设置,在哪里设置?),同时建立一个变量表用来存放执行过程中的变量和值(这个变量表在哪里?) php调用各个模块的rinit方法,即初始化请求,, 比如session扩展中有个RINIT作用的典型示例,如果启用了session.auto_start选项,RINIT将自动触发用户空间的session_start()函数以及预组装$_SESSION变量。 rinit方法可以看做一个准备活动,在程序执行之间就会自动启动 3.一旦请求被初始化了,ZE开始接管控制权,将PHP脚本翻译成符号,最终形成操作码并逐步运行之。如任一操作码需要调用扩展的函数,ZE将会把参数绑定到该函数,并且临时交出控制权直到函数运行结束。 执行完毕 1.当程序执行到末尾或exit或die ,php就会调用清理程序,按顺序调用程序的rshutdown方法,RSHUTDOWN用以清除程序运行时产生的符号表,也就是对每个变量调用unset函数。 一个典型的RSHUTDOWN方法如下: PHP_RSHUTDOWN_FUNCTION(extension_name) { /* Do memory management, unset all variables used in the last PHP call etc */ } 2.所有的请求都已处理完毕,SAPI也准备关闭了,PHP开始执行第二步: PHP调用每个扩展的MSHUTDOWN方法,这是各个模块最后一次释放内存的机会。 一个典型的RSHUTDOWN方法如下: PHP_MSHUTDOWN_FUNCTION(extension_name) { /* Free handlers and persistent memory etc */ } 这样,整个PHP生命周期就结束了。要注意的是,只有在服务器没有请求的情况下才会执行“启动第一步”和“关闭第二步”。