php生命周期
这里介绍的php生命周期,就是php在请求调用过程中,经过了那些操作,都是完成了什么工作。要想了解这个过程,一切就需要从ASPI开始。
SAPI(Server Application Propramming interface)指的是PHP具体应用的编程接口,他提供了一个接口,使得PHP可以和其他应用进行交互数据。PHP脚本要执行可以有很多方式,例如web服务器请求加载(cgi/fast-cgi)和命令行模式执行(cli),也可以嵌入其他程序中。脚本执行的开始都是以SAPI接口实现开始的,只是不同的SAPI接口实现会完成他们特定的工作,例如Apache的mod_php SAPI实现需要初始化从Apache获取的一些信息,在输出内容是将内容返回给Apache,其他的SAPI实现也类似。
下面就是说明SAPI的工作过程:
一、详解一次请求生命周期
我们从未手动开启过PHP的相关进程,它是随着Apache的启动而运行的。PHP通过mod_php5.so模块和Apache相连。PHP总共有三个模块:内核、Zend引擎、以及扩展层:
①PHP内核用来处理请求、文件流、错误处理等先关操作;
②Zend引擎(ZE)用以将源文件转换成机器语言,然后在虚拟机上运行它;
③扩展层是一组函数、类库和流,PHP使用它们来执行一些特殊的操作。
例如:我们需要mysql扩展来连接mysql,当ZE执行程序时可能会需要连接若干扩展,这是ZE将控制权交给扩展,等处理完特定任务后在返还;最后,ZE将程序运行结果返回给PHP内核,它在将结果传送给SAPI层,最终输出到浏览器上。
真正的内部运行过程没有那么简单,上面的例子这是简略的说明了一下。其实Apache启动后,PHP解析器程序也随之启动,PHP开始执行以后会经过两个主要的阶段:
①处理请求的开始阶段。在这个阶段也存在两个过程分别是:第一步MINIT初始化一些环境变量,在整个SAPI生命周期内(例如Apache启动以后的整个生命周期内或者命令行程序整个执行过程中)该过程只进行一次,这将在整个SAPI生命周期中发挥作用;第二步RINIT生成只针对当前请求的一些环境设置。
②请求之后的结束阶段。一般脚本执行到末尾或者通过调用exit()或die()函数,PHP都将进入结束阶段。和开始阶段相对应,结束阶段也分为两个环节,一个在请求结束后停用模块(RSHUTDOWN,对应RINIT),一个在SAPI生命周期结束(WEB服务器退出或者命令行脚本执行完毕退出)时关闭模块(MSHUTDOWN,对应MINIT)。
1、PHP启动第一步MINIT
在这里需要谨记“第一步的操作在任何请求到达之前就发生了”。启动apache后,PHP解析程序也随之启动。PHP调用各个扩展的MINIT方法,从而使这些扩展切换到可用状态(PHP的扩展在php.ini中通过extension关键字引入)。MINIT的意思是“模块初始化”。各个模块都定义了一组函数、类库等用以处理其他请求。一个典型的MINIT方法如下:
PHP_MINIT_FUNCTION(extension_name){ // 注册常量或者类等初始化操作 return SUCCESS; }
2、php启动第二步RINIT
当一个页面请求发生时,SAPI层将控制权交给PHP层。于是PHP设置了用户回复本次请求所需的环境变量。同时,他还建立一个环境表,用来存放执行过程中产生的变量名和值。PHP调用各个模块的RINIT方法,即“请求初始化”。一个经典的例子是session模块的RINIT,如果在php.ini中启用了session模块,那在调用该模块的RINIT时会初始化$_SESSION变量,并将相关内容读入;一个典型的RINIT方法如下:
PHP_RINIT_FUNCTION(extension_name){ // 例如记录请求开始时间 // 随后在请求结束的时候记录结束时间。这样我们就能够记录下处理请求所花费的时间了 return SUCCESS; }
3、关闭第一步RSHUTDOWN
一旦页面执行完毕,php就会启动清理程序。它会按顺序调用各个模块的RSHUTDOWN方法。RSHUTDOWN用以清理程序运行时产生的符号表,也就是对每个变量调用unset函数。一个典型的RSHUTDOWN方法如下:
PHP_RSHUTDOWN_FUNCTION(extension_name){ // 例如记录请求结束时间,并把相应的信息写入到日至文件中。 return SUCCESS; }
4、关闭第二步MSHUTDOWN
如果web服务器关闭(例如Apache关闭),SAPI也就准备关闭了,php开始执行第二步:php调用各个扩展的MSHUTDOWN方法,这是各个模块最后一次释放内存的机会。一个典型的MSHUTDOWN方法如下:
PHP_MSHUTDOWN_FUNCTION(extension_name) { /* Free handlers and persistent memory etc */ }
下面是一种展示整个过程的图片: