1、PHP的运行模式 ) n4 _2 a' b* X# `0 _+ d PHP两种运行模式是WEB模式、CLI模式。无论哪种模式,PHP工作原理都是一样的,作为一种SAPI运行。% Z m7 f/ H0 T+ u& E( S a) t. s
1、当我们在终端敲入php这个命令的时候,它使用的是CLI。 $ c t, i' K) o$ H/ p" P 它就像一个web服务器一样来支持php完成这个请求,请求完成后再重新把控制权交给终端。 4 B o3 p: X1 D3 _2、当使用Apache或者别web服务器作为宿主时,当一个请求到来时,PHP会来支持完成这个请求。一般有: 6 W3 w( d% a1 _) r
多进程(通常编译为apache的模块来处理PHP请求)
多线程模式8 M2 u+ X' z9 Q9 ]* {% A3 _% i
2、一切的开始: SAPI接口( K: ~# e) N" K, S
通常我们编写php Web程序都是通过Apache或者Nginx这类Web服务器来测试脚本. 或者在命令行下通过php程序来执行PHP脚本. 执行完成脚本后,服务器应答,浏览器显示应答信息,或者在命令结束后在标准输出显示内容. 我们很少关心PHP解释器在哪里. 虽然通过Web服务器和命令行程序执行脚本看起来很不一样. 实际上她们的工作是一样的. 命令行程序和Web程序类似, 命令行参数传递给要执行的脚本,相当于通过url 请求一个PHP页面. 脚本戳里完成后返回响应结果,只不过命令行响应的结果是显示在终端上. 脚本执行的开始都是通过SAPI接口进行的. ' D7 _' k F* F6 Z8 X 启动apache + |. P/ L0 N. l5 e( X/ [ 当给定的SAPI启动时,例如在对/usr/local/apache/bin/apachectl start的响应中,PHP由初始化其内核子系统开始。在接近启动例程的末尾,它加载每个扩展的代码并调用其模块初始化例程(MINIT)。这使得每个扩展可以初始化内部变量、分配资源、注册资源处理器,以及向ZE注册自己的函数,以便于脚本调用这其中的函数时候ZE知道执行哪些代码。 . G, T: z" ^% e4 F8 o: J请求处理初始化% }. S( o! a: x/ d0 U/ ]% a
接下来,PHP等待SAPI层请求要处理的页面。对于CGI或CLI等SAPI,这将立刻发生且只发生一次。对于Apache、IIS或其他成熟的web服务器SAPI,每次远程用户请求页面时都将发生,因此重复很多次,也可能并发。不管请求如何产生,PHP开始于要求ZE建立脚本的运行环境,然后调用每个扩展的请求初始化 (RINIT)函数。RINIT使得扩展有机会设定特定的环境变量,根据请求分配资源,或者执行其他任务,如审核。 session扩展中有个RINIT作用的典型示例,如果启用了session.auto_start选项,RINIT将自动触发用户空间的session_start()函数以及预组装$_SESSION变量。 + Z# k* `) a2 s* [, F执行php代码7 \2 q6 T% }/ e, G* X# B# B* k
一旦请求被初始化了,ZE开始接管控制权,将PHP脚本翻译成符号,最终形成操作码并逐步运行之。如任一操作码需要调用扩展的函数,ZE将会把参数绑定到该函数,并且临时交出控制权直到函数运行结束。" T) ?6 L! q. {1 l. w2 ]0 v 脚本结束 3 F. |3 i" a6 i8 h 脚本运行结束后,PHP调用每个扩展的请求关闭(RSHUTDOWN)函数以执行最后的清理工作(如将session变量存入磁盘)。接下来,ZE执行清理过程(垃圾收集)-有效地对之前的请求期间用到的每个变量执行unset()。( e: o, l5 r! s sapi关闭6 m) ` P5 [; V+ Y
一旦完成,PHP继续等待SAPI的其他文档请求或者是关闭信号。对于CGI和CLI等SAPI,没有“下一个请求”,所以SAPI立刻开始关闭。关闭期间,PHP再次遍历每个扩展,调用其模块关闭(MSHUTDOWN)函数,并最终关闭自己的内核子系统。 v( y8 M% k' y9 m$ ^. b+ {简要的过程如下: 0 Z! _- o$ Q5 z$ a8 B0 d$ f( {1. PHP是随着Apache的启动而运行的; o& r- C' S S- c+ r# x
2. PHP通过mod_php5.so模块和Apache相连(具体说来是SAPI,即服务器应用程序编程接口);/ B/ U. p! j/ u, N' V! d; H5 P
3. PHP总共有三个模块:内核、Zend引擎、以及扩展层;* M2 Q" i! O# {4 Z. _
4. PHP内核用来处理请求、文件流、错误处理等相关操作;" ~9 p$ e, Y w# T X" d
5. Zend引擎(ZE)用以将源文件转换成机器语言,然后在虚拟机上运行它; 8 S" o4 }: r! z7 D6. 扩展层是一组函数、类库和流,PHP使用它们来执行一些特定的操作。比如,我们需要mysql扩展来连接MySQL数据库; ) x9 a% M8 S! P! F% _* f7. 当ZE执行程序时可能会需要连接若干扩展,这时ZE将控制权交给扩展,等处理完特定任务后再返还; 0 u0 \7 f0 w- z5 U, Q# s8. 最后,ZE将程序运行结果返回给PHP内核,它再将结果传送给SAPI层,最终输出到浏览器上。 / `+ S! F3 s+ I$ G, J3、PHP的开始和结束阶段+ D# G/ _/ T, r; _# H; j8 q
开始阶段有两个过程: / v% C2 u) G5 f) k! s3 a: M7 c 第一个过程:apache启动的过程,即在任何请求到达之前就发生。是在整个SAPI生命周期内(例如Apache启动以后的整个生命周期内或者命令行程序整个执行过程中)的开始阶段(MINIT),该阶段只进行一次.。启动Apache后,PHP解释程序也随之启动; PHP调用各个扩展(模块)的MINIT方法,从而使这些扩展切换到可用状态。看看php.ini文件里打开了哪些扩展吧; MINIT的意思是“模块初始化”。各个模块都定义了一组函数、类库等用以处理其他请求。 模块在这个阶段可以进行一些初始化工作,例如注册常量, 定义模块使用的类等等.典型的的模块回调函数MINIT方法如下: % ]2 G. p3 _ T; W$ A$ _
结束阶段分为两个环节: 请求处理完后就进入了结束阶段, 一般脚本执行到末尾或者通过调用exit()或者die()函数,PHP都将进入结束阶段. 和开始阶段对应,结束阶段也分为两个环节,一个在请求结束后(RSHUWDOWN),一个在SAPI生命周期结束时(MSHUTDOWN). - ^# L M' q, E' l) f$ M: w6 W6 O第一个环节:请求处理完后结束阶段:请求处理完后就进入了结束阶段,PHP就会启动清理程序。它会按顺序调用各个模块的RSHUTDOWN方法。 RSHUTDOWN用以清除程序运行时产生的符号表,也就是对每个变量调用unset函数。典型的RSHUTDOWN方法如下: 1 j. c) e7 Z( I8 [5 U
第二个环节:最后,所有的请求都已处理完毕,SAPI也准备关闭了, PHP调用每个扩展的MSHUTDOWN方法,这是各个模块最后一次释放内存的机会。(这个是对于CGI和CLI等SAPI,没有“下一个请求”,所以SAPI立刻开始关闭。)典型的RSHUTDOWN方法如下:5 U2 f8 m" Z% k) f' O4 r