简介

  先看看下面这个过程:

  • 我们从未手动开启过PHP的相关进程,它是随着Apache的启动而运行的;

  • PHP通过mod_php5.so模块和Apache相连(具体说来是SAPI,即服务器应用程序编程接口);

  • PHP总共有三个模块:内核、Zend引擎、以及扩展层;

  • PHP内核用来处理请求、文件流、错误处理等相关操作;

  • Zend引擎(ZE)用以将源文件转换成机器语言,然后在虚拟机上运行它;

  • 扩展层是一组函数、类库和流,PHP使用它们来执行一些特定的操作。比如,我们需要mysql扩展来连接MySQL数据库;

  • 当ZE执行程序时可能会需要连接若干扩展,这时ZE将控制权交给扩展,等处理完特定任务后再返还;

  • 最后,ZE将程序运行结果返回给PHP内核,它再将结果传送给SAPI层,最终输出到浏览器上。

  深入探讨

  等等,没有这么简单。以上过程只是个简略版,让我们再深入挖掘一下,看看幕后还发生了些什么。

  • Apache启动后,PHP解释程序也随之启动;

  • PHP的启动过程有两步;

  • 第一步是初始化一些环境变量,这将在整个SAPI生命周期中发生作用;

  • 第二步是生成只针对当前请求的一些变量设置。

  PHP启动第一步

  不清楚什么第一第二步是什么?别担心,我们接下来详细讨论一下。让我们先看看第一步,也是最主要的一步。要记住的是,第一步的操作在任何请求到达之前就发生了。

  • 启动Apache后,PHP解释程序也随之启动;

  • PHP调用各个扩展的MINIT方法,从而使这些扩展切换到可用状态。看看php.ini文件里打开了哪些扩展吧;

  • MINIT的意思是“模块初始化”。各个模块都定义了一组函数、类库等用以处理其他请求。

  一个典型的MINIT方法如下:

PHP代码
  1. PHP_MINIT_FUNCTION(extension_name){
  2. /* Initialize functions, classes etc */
  3. }

  PHP启动第二步

  • 当一个页面请求发生时,SAPI层将控制权交给PHP层。于是PHP设置了用于回复本次请求所需的环境变量。同时,它还建立一个变量表,用来存放执行过程中产生的变量名和值。

  • PHP调用各个模块的RINIT方法,即“请求初始化”。一个经典的例子是Session模块的RINIT,如果在php.ini中启用了Session模块,那在调用该模块的RINIT时就会初始化$_SESSION变量,并将相关内容读入;

  • RINIT方法可以看作是一个准备过程,在程序执行之间就会自动启动。

  一个典型的RINIT方法如下:

PHP代码
  1. PHP_RINIT_FUNCTION(extension_name) {
  2. /* Initialize session variables, pre-populate variables, redefine global variables etc */
  3. }

  PHP关闭第一步

  如同PHP启动一样,PHP的关闭也分两步:

  • 一旦页面执行完毕(无论是执行到了文件末尾还是用exit或die函数中止),PHP就会启动清理程序。它会按顺序调用各个模块的RSHUTDOWN方法。

  • RSHUTDOWN用以清除程序运行时产生的符号表,也就是对每个变量调用unset函数。

  一个典型的RSHUTDOWN方法如下:

PHP代码
  1. PHP_RSHUTDOWN_FUNCTION(extension_name) {
  2. /* Do memory management, unset all variables used in the last PHP call etc */
  3. }

  PHP关闭第二步

  最后,所有的请求都已处理完毕,SAPI也准备关闭了,PHP开始执行第二步:

  • PHP调用每个扩展的MSHUTDOWN方法,这是各个模块最后一次释放内存的机会。

  一个典型的RSHUTDOWN方法如下:

PHP代码
  1. PHP_MSHUTDOWN_FUNCTION(extension_name) {
  2. /* Free handlers and persistent memory etc */
  3. }

  这样,整个PHP生命周期就结束了。要注意的是,只有在服务器没有请求的情况下才会执行“启动第一步”和“关闭第二步”。

  下面的是用一些图示来说明的!

  PHP底层工作原理

图1 php结构

  从图上可以看出,php从下到上是一个4层体系

  ①Zend引擎

  Zend整体用纯c实现,是php的内核部分,它将php代码翻译(词法、语法解析等一系列编译过程)为可执行opcode的处理并实现相应的处理方法、实现了基本的数据结构(如hashtable、oo)、内存分配及管理、提供了相应的api方法供外部调用,是一切的核心,所有的外围功能均围绕zend实现。

  ②Extensions

  围绕着zend引擎,extensions通过组件式的方式提供各种基础服务,我们常见的各种内置函数(如array系列)、标准库等都是通过extension来实现,用户也可以根据需要实现自己的extension以达到功能扩展、性能优化等目的(如贴吧正在使用的php中间层、富文本解析就是extension的典型应用)。

  ③Sapi

  Sapi全称是Server Application Programming Interface,也就是服务端应用编程接口,sapi通过一系列钩子函数,使得php可以和外围交互数据,这是php非常优雅和成功的一个设计,通过sapi成功的将php本身和上层应用解耦隔离,php可以不再考虑如何针对不同应用进行兼容,而应用本身也可以针对自己的特点实现不同的处理方式。后面将在sapi章节中介绍

  ④上层应用

  这就是我们平时编写的php程序,通过不同的sapi方式得到各种各样的应用模式,如通过webserver实现web应用、在命令行下以脚本方式运行等等。

  构架思想:

  引擎(Zend)+组件(ext)的模式降低内部耦合

  中间层(sapi)隔绝web server和php

  **************************************************************************

  如果php是一辆车,那么

  车的框架就是php本身

  Zend是车的引擎(发动机)

  Ext下面的各种组件就是车的轮子

  Sapi可以看做是公路,车可以跑在不同类型的公路上

  而一次php程序的执行就是汽车跑在公路上。

  因此,我们需要:性能优异的引擎+合适的车轮+正确的跑道

  Apache和php的关系

  Apache对于php的解析,就是通过众多Module中的php Module来完成的。

  把php最终集成到Apache系统中,还需要对Apache进行一些必要的设置。这里,我们就以php的mod_php5 SAPI运行模式为例进行讲解,至于SAPI这个概念后面我们还会详细讲解。

  假定我们安装的版本是Apache2 和 Php5,那么需要编辑Apache的主配置文件http.conf,在其中加入下面的几行内容:

  Unix/Linux环境下:

  LoadModule php5_module modules/mod_php5.so

  AddType application/x-httpd-php .php

  注:其中modules/mod_php5.so 是X系统环境下mod_php5.so文件的安装位置。

  Windows环境下:

  LoadModule php5_module d:/php/php5apache2.dll

  AddType application/x-httpd-php .php

  注:其中d:/php/php5apache2.dll 是在Windows环境下php5apache2.dll文件的安装位置。

  这两项配置就是告诉Apache Server,以后收到的Url用户请求,凡是以php作为后缀,就需要调用php5_module模块(mod_php5.so/ php5apache2.dll)进行处理。

  Apache的生命周期

  Apach的请求处理流程

  Apache请求处理循环详解

  Apache请求处理循环的11个阶段都做了哪些事情呢?

  1、Post-Read-Request阶段

  在正常请求处理流程中,这是模块可以插入钩子的第一个阶段。对于那些想很早进入处理请求的模块来说,这个阶段可以被利用。

  2、URI Translation阶段

  Apache在本阶段的主要工作:将请求的URL映射到本地文件系统。模块可以在这阶段插入钩子,执行自己的映射逻辑。mod_alias就是利用这个阶段工作的。

  3、Header Parsing阶段

  Apache在本阶段的主要工作:检查请求的头部。由于模块可以在请求处理流程的任何一个点上执行检查请求头部的任务,因此这个钩子很少被使用。mod_setenvif就是利用这个阶段工作的。

  4、Access Control阶段

  Apache在本阶段的主要工作:根据配置文件检查是否允许访问请求的资源。Apache的标准逻辑实现了允许和拒绝指令。mod_authz_host就是利用这个阶段工作的。

  5、Authentication阶段

  Apache在本阶段的主要工作:按照配置文件设定的策略对用户进行认证,并设定用户名区域。模块可以在这阶段插入钩子,实现一个认证方法。

  6、Authorization阶段

  Apache在本阶段的主要工作:根据配置文件检查是否允许认证过的用户执行请求的操作。模块可以在这阶段插入钩子,实现一个用户权限管理的方法。

  7、MIME Type Checking阶段

  Apache在本阶段的主要工作:根据请求资源的MIME类型的相关规则,判定将要使用的内容处理函数。标准模块mod_negotiation和mod_mime实现了这个钩子。

  8、FixUp阶段

  这是一个通用的阶段,允许模块在内容生成器之前,运行任何必要的处理流程。和Post_Read_Request类似,这是一个能够捕获任何信息的钩子,也是最常使用的钩子。

  9、Response阶段

  Apache在本阶段的主要工作:生成返回客户端的内容,负责给客户端发送一个恰当的回复。这个阶段是整个处理流程的核心部分。

  10、Logging阶段

  Apache在本阶段的主要工作:在回复已经发送给客户端之后记录事务。模块可能修改或者替换Apache的标准日志记录。

  11、CleanUp阶段

  Apache在本阶段的主要工作:清理本次请求事务处理完成之后遗留的环境,比如文件、目录的处理或者Socket的关闭等等,这是Apache一次请求处理的最后一个阶段。

  LAMP架构:

  从下往上四层:

  ①liunx 属于操作系统的底层

  ②apache服务器,属于次服务器,沟通linux和PHP

  ③php:属于服务端编程语言,通过php_module 模块 和apache关联

  ④mysql和其他web服务:属于应用服务,通过PHP的Extensions外 挂模块和mysql关联

转载:http://www.jizhuomi.com/software/662.html

详解PHP的执行原理和流程的更多相关文章

  1. [转帖]万字详解Oracle架构、原理、进程,学会世间再无复杂架构

    万字详解Oracle架构.原理.进程,学会世间再无复杂架构 http://www.itpub.net/2019/04/24/1694/ 里面的图特别好 数据和云 2019-04-24 09:11:59 ...

  2. 第三节:带你详解Java的操作符,控制流程以及数组

    前言 大家好,给大家带来带你详解Java的操作符,控制流程以及数组的概述,希望你们喜欢 操作符 算数操作符 一般的 +,-,*,/,还有两个自增 自减 ,以及一个取模 % 操作符. 这里的操作算法,一 ...

  3. SpringCloud 详解配置刷新的原理 使用jasypt自动加解密后 无法使用 springcloud 中的自动刷新/refresh功能

    之所以会查找这篇文章,是因为要解决这样一个问题: 当我使用了jasypt进行配置文件加解密后,如果再使用refresh 去刷新配置,则自动加解密会失效. 原因分析:刷新不是我之前想象的直接调用conf ...

  4. RocketMQ详解(一)原理概览

    专题目录 RocketMQ详解(一)原理概览 RocketMQ详解(二)安装使用详解 RocketMQ详解(三)启动运行原理 RocketMQ详解(四)核心设计原理 RocketMQ详解(五)总结提高 ...

  5. 淘宝JAVA中间件Diamond详解(2)-原理介绍

    淘宝JAVA中间件Diamond详解(二)---原理介绍 大家好,通过第一篇的快速使用,大家已经对diamond有了一个基本的了解.本次为大家带来的是diamond核心原理的介绍,主要包括server ...

  6. [转帖]详解Linux系统inode原理--硬链接、软链接、innodb大小和划分等

    详解Linux系统inode原理--硬链接.软链接.innodb大小和划分等 原创 波波说运维 2019-07-17 00:03:00 https://www.toutiao.com/i6713116 ...

  7. sed 增删改查详解以及 sed -i原理

    我为什么要详细记录sed命令:     sed 擅长取行.工作中三剑客使用频率最高,本篇文章将对sed命令常用的 增,删,改,查 进行详细讲解,以备以后工作中遗忘了查询,sed命令是作为运维人员来说, ...

  8. JDBC详解系列(一)之流程

    ---[来自我的CSDN博客](http://blog.csdn.net/weixin_37139197/article/details/78838091)--- JDBC概述   使用JDBC也挺长 ...

  9. Android中measure过程、WRAP_CONTENT详解以及 xml布局文件解析流程浅析

    转自:http://www.uml.org.cn/mobiledev/201211221.asp 今天,我着重讲解下如下三个内容: measure过程 WRAP_CONTENT.MATCH_PAREN ...

随机推荐

  1. 在ubuntu下运行python脚本

    转自http://www.cnblogs.com/hester/p/5575658.html 1. 运行方式一 新建test.py文件: 1 touch test.py 然后vim test.py打开 ...

  2. PAT 甲级 1117 Eddington Number

    https://pintia.cn/problem-sets/994805342720868352/problems/994805354762715136 British astronomer Edd ...

  3. 查看django版本的方法

    在cmd输入: python -m django --version django-admin --version

  4. MT【74】不可能是哪个函数?

    解答:设$f(g(x_0))=x_0$,则$g(f(g(x_0)))=g(x_0)$,令$y_0=g(x_0)$则$g(f(y_0))=y_0$有解.易得答案为$B$.

  5. 【刷题】LOJ 6002 「网络流 24 题」最小路径覆盖

    题目描述 给定有向图 \(G = (V, E)\) .设 \(P\) 是 \(G\) 的一个简单路(顶点不相交)的集合.如果 \(V\) 中每个顶点恰好在 \(P\) 的一条路上,则称 \(P\) 是 ...

  6. 界面编程之QT的信号与槽20180725

    /*******************************************************************************************/ 一.指定父对 ...

  7. 在windows上部署使用Redis出现问题的解决方法

    下载Redis 在Redis的官网下载页上有各种各样的版本,我这次是在windows上部署的,要去GitHub上下载.目前的是2.8.12版的,直接解压,在\bin\release 目录下有个压缩包, ...

  8. DCT变换、DCT反变换、分块DCT变换

    一.引言 DCT变换的全称是离散余弦变换(Discrete Cosine Transform),主要用于将数据或图像的压缩,能够将空域的信号转换到频域上,具有良好的去相关性的性能.DCT变换本身是无损 ...

  9. 将句子表示为向量(下):基于监督学习的句子表示学习(sentence embedding)

    1. 引言 上一篇介绍了如何用无监督方法来训练sentence embedding,本文将介绍如何利用监督学习训练句子编码器从而获取sentence embedding,包括利用释义数据库PPDB.自 ...

  10. linux command ------ netstat

    netstat命令是一个监控TCP/IP网络的非常有用的工具,它可以显示路由表.实际的网络连接以及每一个网络接口设备的状态信息. 语法选项 netstat [选项] -a或--all:显示所有连线中的 ...