PHP高級之一次請求處理過程或生命周期詳解

2020-12-15 迷人主題

簡介先看看下面這個過程:

我們從未手動開啟過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_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生命周期就結束了。要注意的是,只有在伺服器沒有請求的情況下才會執行「啟動第一步」和「關閉第二步」。

圖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程序的執行就是汽車跑在公路上。

因此,我們需要:性能優異的引擎+合適的車輪+正確的跑道

把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請求處理循環詳解 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一次請求處理的最後一個階段。

相關焦點

  • 圖解 Laravel 請求的完整生命周期
    PHP 的生命周期生命周期當我們請求一個php文件時,PHP 為了完成這次請求,會發生5個階段的生命周期切換:模塊初始化(MINIT),即調用 php.ini 中指明的擴展的初始化函數進行初始化工作,如 mysql 擴展。
  • 手把手擼PHP擴展 0x03: 理解PHP生命周期的過程
    PHP_MSHUTDOWN_FUNCTION(study){ return SUCCESS;}PHP_RINIT_FUNCTION(study){ return SUCCESS;}PHP_RSHUTDOWN_FUNCTION(study){ return SUCCESS;}這幾個函數是伴隨著PHP生命周期來執行的
  • PHP 垃圾回收機制詳解
    PHP strtotime應用經驗之談PHP memory_get_usage()管理內存PHP unset全局變量運用問題詳解PHP unset()函數銷毀變量教你快速實現PHP全站權限驗證一、PHP 垃圾回收機制(Garbage Collector 簡稱GC) 在PHP中,沒有任何變量指向這個對象時,這個對象就成為垃圾。
  • PHP 後端表單驗證和請求處理
    phpnamespace App\Model;use Illuminate\Database\Eloquent\Model;class Message extends Model{    public $timestamps = false;}表單數據處理邏輯做好上述準備後,接下來,我們在 HomeController
  • PHP中__get()方法詳解
    相關:《__construct(),類的構造函數》《__destruct(),類的析構函數》《__call()方法詳解》《__callStatic()方法詳解》PHP中__get(),獲得一個類的成員變量時調用在 php 面向對象編程中,類的成員屬性被設定為 private
  • 「Find」php的GET請求網址
    Hello,大家好,今天有了一個想法,但是之前沒有學習過php,所以在網上搜索了一下,就是網頁GET請求構造網址,處理返回的JSON數據。<?php$url="網址";$html = file_get_contents($url);echo $html;//這樣網頁上應該就顯示你訪問網頁返回的內容了?>如果你跟我一樣希望處理JSON數據,可以用到json_decode()<?
  • php中Session使用方法詳解
    Session以數組的形式使用,如:$_SESSION['session名']註冊一個會話變量和讀取Session  在PHP中使用Session變量,除了要啟動之外,還要經過註冊的過程。註冊和讀取Session變量,都要通過訪問$_SESSION數組完成。在$_SESSION關聯數組中的鍵名具有和PHP中普通變量相同的命名規則。
  • PHP 使用 CURL 詳解
    下面的程序片段是使用CURL發送HTTP的典型過程<?CURL的選項還有很多,可以到PHP的官方網站(http://www.php.net/manual/en/function.curl-setopt.php)上查看CURL支持的所有選項列表。
  • 第285天:React生命周期
    React生命周期React的生命周期從廣義上分為掛載、渲染、卸載三個階段,在React的整個生命周期中提供很多鉤子函數在生命周期的不同時刻調用
  • php session 會話(專題)
    >4、使用 session 實現登錄功能對於 Cookie 來說,假設我們要驗證用戶是否登陸,就必須在 Cookie 中保存用戶名和密碼(可能是 md5 加密後字符串),並在每次請求頁面的時候進行驗證。如果用戶名和密碼存儲在資料庫,每次都要執行一次資料庫查詢,給資料庫造成多餘的負擔。因為我們並不能 只做一次驗證。
  • ELF PHP 可執行程序運行後加載重型腳本的過程
    4 啟動服務nginx只是轉發請求給php-fpm處理而已,畢竟這是老傳統了,當你願意遵守這個傳統條款時。7 接下來,我們一步步的看一下我運行,php-fpm為了處理這個重型框架到底加載了多少php腳本文件【部分截圖】【自己用電腦看,不要用小的可憐兮兮的手機看,渣男就應該用PC看】
  • servlet生命周期一共分幾步?
    Servlet生命周期分為三個階段:初始化階段、響應客戶請求階段、終止階段,其具體操作如下圖所示:首先簡單解釋一下Servlet接收和響應客戶請求的過程,首先客戶發送一個請求,Servlet是調用service()方法對請求進行響應 的,通過原始碼可見
  • servlet生命周期
    1.servlet是運行在服務端的java程序2.servlet的生命周期主要有三個方法:init()初始化階段service()處理客戶端請求階段destroy()終止階段初始化階段:Servlet容器加載Servlet,加載完成後,Servlet容器會創建一個Servlet實例並調用init()方法
  • PHP運行方式介紹
    請求結束後銷毀進程,所以性能比較低。Fast-CGI 協議模式Fast-CGI 是 CGI 模式的升級版,它像是一個常駐內存的 CGI,只要開啟後,就可以一直處理請求,不再需要結束進程。調用過程如下:Web伺服器 Fast-CGI 進程管理器初始化 -> 預先fork多個進程用戶請求 -> Web伺服器接收請求 ->Web伺服器將請求交給 Fast-CGI 進程管理器 ->Fast-CGI 進程管理器接收,給其中一個空閒的的
  • 我對 React V16.4 生命周期的理解
    React 為三個生命周期函數加上了 UNSAFE 前綴,並明確表示會在 V17.0 版本中刪除這三個生命周期函數。而且 UNSAFE_componentWillMount 在伺服器端渲染也會被調用到(此方法是服務端渲染唯一會調用的生命周期函數),你肯定不希望 AJAX 請求被執行多次,所以這樣的 IO 操作放在 componentDidMount 中更合適。
  • PHP在Linux下執行定時任務的實現思路詳解
    確定,這樣一個定時任務就建立好了,在這個定時任務上右鍵,運行,這個定時任務就開始執行了,到點時,就會運行cron.bat處理,cron.bat再去執行php。$loop : $run_time - $gmt_time; // 這裡處理是為了確定還要等多久才開始第一次執行任務,$loop就是要等多久才執行的時間間隔$loop = $loop > 0 ? $loop : 0; if(!
  • php mysql PDO 查詢操作的實例詳解
    http://www.jb51.net/article/124388.htm這篇文章主要介紹了php mysql PDO
  • PHP 實現簡單的 MVC 框架
    主要處理前臺模塊的操作model/:前臺模型目錄,存放模型文件。處理前臺模型的相關操作view/:前臺視圖目錄,存放視圖文件。前臺展示的模板文件。config/:配置文件目錄config.php:框架的配置文件runtime/:運行時目錄,保存框架運行時產生的數據。cache/:緩存目錄。
  • php面試題之—PHP核心技術(高級部分)
    >相關題目:使用五種以上方式獲取一個文件的擴展名,要求:dir/upload.image.jpg,找出.jpg或者jpg,必須使用PHP自帶的處理函數進行處理,方法不能明顯重複,可以封裝成函數,比如get_ext1(filename),getext2(file_name)8.寫一個函數,能夠遍歷一個文件夾下的所有文件和子文件夾。
  • 詳解服務體驗生命周期,助你提升用戶體驗
    這個「生命周期」是怎麼回事?實際上所有的服務體驗都遵循一個相同的生命周期。二、實踐篇1. 客戶 vs 服務的生命周期當我們談論「生命周期」時,我們說的是隨著時間的推移而發生的「事件」或「階段」的端到端序列。