PHP的執(zhí)行原理/執(zhí)行流程

更深入的學(xué)習(xí)和了解可以查看下面:
風(fēng)雨的博客http://www.laruence.com/2008/08/12/180.html
百度研發(fā)中心的博客http://stblog.baidu-tech.com/?p=763
王興賓的博客http://blog.csdn.net/wanghao72214/article/details/3916825

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

  • 我們從未手動開啟過PHP的相關(guān)進(jìn)程,它是隨著Apache的啟動而運行的;
  • PHP通過mod_php5.so模塊和+ + + Apache相連(具體說來是SAPI,即服務(wù)器應(yīng)用程序編程接口)浊猾;
  • PHP總共有三個模塊:內(nèi)核蔫磨、Zend引擎慨代、以及擴(kuò)展層阵面;
  • PHP內(nèi)核用來處理請求、文件流梧却、錯誤處理等相關(guān)操作;
  • Zend引擎(ZE)用以將源文件轉(zhuǎn)換成機(jī)器語言败去,然后在虛擬機(jī)上運行它篮幢;
  • 擴(kuò)展層是一組函數(shù)、類庫和流为迈,PHP使用它們來執(zhí)行一些特定的操作三椿。比如缺菌,我們需要mysql擴(kuò)展來連接MySQL數(shù)據(jù)庫;
  • 當(dāng)ZE執(zhí)行程序時可能會需要連接若干擴(kuò)展搜锰,這時ZE將控制權(quán)交給擴(kuò)展伴郁,等處理完特定任務(wù)后再返還;
  • 最后蛋叼,ZE將程序運行結(jié)果返回給PHP內(nèi)核焊傅,它再將結(jié)果傳送給SAPI層,最終輸出到瀏覽器上狈涮。

深入探討  
等等狐胎,沒有這么簡單。以上過程只是個簡略版歌馍,讓我們再深入挖掘一下握巢,看看幕后還發(fā)生了些什么。

  • Apache啟動后松却,PHP解釋程序也隨之啟動暴浦;
  • PHP的啟動過程有兩步;
  • 第一步是初始化一些環(huán)境變量晓锻,這將在整個SAPI生命周期中發(fā)生作用歌焦;
  • 第二步是生成只針對當(dāng)前請求的一些變量設(shè)置。

PHP啟動第一步  
不清楚什么第一第二步是什么砚哆?別擔(dān)心独撇,我們接下來詳細(xì)討論一下。讓我們先看看第一步躁锁,也是最主要的一步券勺。要記住的是,第一步的操作在任何請求到達(dá)之前就發(fā)生了灿里。

  • 啟動Apache后关炼,PHP解釋程序也隨之啟動;
  • PHP調(diào)用各個擴(kuò)展的MINIT方法匣吊,從而使這些擴(kuò)展切換到可用狀態(tài)儒拂。看看php.ini文件里打開了哪些擴(kuò)展吧色鸳;
  • MINIT的意思是“模塊初始化”社痛。各個模塊都定義了一組函數(shù)、類庫等用以處理其他請求命雀。
      一個典型的MINIT方法如下:
PHP_MINIT_FUNCTION(extension_name){
/* Initialize functions, classes etc */
}

PHP啟動第二步
當(dāng)一個頁面請求發(fā)生時蒜哀,SAPI層將控制權(quán)交給PHP層。于是PHP設(shè)置了用于回復(fù)本次請求所需的環(huán)境變量吏砂。同時撵儿,它還建立一個變量表乘客,用來存放執(zhí)行過程中產(chǎn)生的變量名和值。
PHP調(diào)用各個模塊的RINIT方法淀歇,即“請求初始化”易核。一個經(jīng)典的例子是Session模塊的RINIT,如果在php.ini中啟用了Session模塊浪默,那在調(diào)用該模塊的RINIT時就會初始化$_SESSION變量牡直,并將相關(guān)內(nèi)容讀入;
RINIT方法可以看作是一個準(zhǔn)備過程纳决,在程序執(zhí)行之間就會自動啟動碰逸。

一個典型的RINIT方法如下:

PHP_RINIT_FUNCTION(extension_name) 
{/* Initialize session variables, pre-populate variables, redefine global variables etc */}
**PHP關(guān)閉第一步**

如同PHP啟動一樣,PHP的關(guān)閉也分兩步:
一旦頁面執(zhí)行完畢(無論是執(zhí)行到了文件末尾還是用exit或die函數(shù)中止)阔加,PHP就會啟動清理程序饵史。它會按順序調(diào)用各個模塊的RSHUTDOWN方法。
RSHUTDOWN用以清除程序運行時產(chǎn)生的符號表掸哑,也就是對每個變量調(diào)用unset函數(shù)约急。

一個典型的RSHUTDOWN方法如下:

PHP_RSHUTDOWN_FUNCTION(extension_name) {/* Do memory management, unset all variables used in the last PHP call etc */}**PHP關(guān)閉第二步**
```
最后零远,所有的請求都已處理完畢苗分,SAPI也準(zhǔn)備關(guān)閉了,PHP開始執(zhí)行第二步:
PHP調(diào)用每個擴(kuò)展的MSHUTDOWN方法牵辣,這是各個模塊最后一次釋放內(nèi)存的機(jī)會摔癣。

一個典型的RSHUTDOWN方法如下:
```
PHP_MSHUTDOWN_FUNCTION(extension_name) {/* Free handlers and persistent memory etc */}
```  

這樣,整個PHP生命周期就結(jié)束了纬向。要注意的是择浊,只有在服務(wù)器沒有請求的情況下才會執(zhí)行“啟動第一步”和“關(guān)閉第二步”。
 
下面的是用一些圖示來說明的逾条!

![](http://upload-images.jianshu.io/upload_images/2665779-d9e731a7bb71f507.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

圖1 php結(jié)構(gòu)
從圖上可以看出琢岩,php從下到上是一個4層體系
①Zend引擎
Zend整體用純c實現(xiàn),是php的內(nèi)核部分师脂,它將php代碼翻譯(詞法担孔、語法解析等一系列編譯過程)為可執(zhí)行opcode的處理并實現(xiàn)相應(yīng)的處理方法、實現(xiàn)了基本的數(shù)據(jù)結(jié)構(gòu)(如hashtable吃警、oo)糕篇、內(nèi)存分配及管理、提供了相應(yīng)的api方法供外部調(diào)用酌心,是一切的核心拌消,所有的外圍功能均圍繞zend實現(xiàn)。
②Extensions
圍繞著zend引擎安券,extensions通過組件式的方式提供各種基礎(chǔ)服務(wù)墩崩,我們常見的各種內(nèi)置函數(shù)(如array系列)氓英、標(biāo)準(zhǔn)庫等都是通過extension來實現(xiàn),用戶也可以根據(jù)需要實現(xiàn)自己的extension以達(dá)到功能擴(kuò)展泰鸡、性能優(yōu)化等目的(如貼吧正在使用的php中間層债蓝、富文本解析就是extension的典型應(yīng)用)。
③Sapi
Sapi全稱是Server Application Programming Interface盛龄,也就是服務(wù)端應(yīng)用編程接口饰迹,sapi通過一系列鉤子函數(shù),使得php可以和外圍交互數(shù)據(jù)余舶,這是php非常優(yōu)雅和成功的一個設(shè)計啊鸭,通過sapi成功的將php本身和上層應(yīng)用解耦隔離,php可以不再考慮如何針對不同應(yīng)用進(jìn)行兼容匿值,而應(yīng)用本身也可以針對自己的特點實現(xiàn)不同的處理方式赠制。后面將在sapi章節(jié)中介紹
④上層應(yīng)用
這就是我們平時編寫的php程序,通過不同的sapi方式得到各種各樣的應(yīng)用模式挟憔,如通過webserver實現(xiàn)web應(yīng)用钟些、在命令行下以腳本方式運行等等。

**構(gòu)架思想:**
引擎(Zend)+組件(ext)的模式降低內(nèi)部耦合
中間層(sapi)隔絕web server和php
**************************************************************************
如果php是一輛車绊谭,那么
車的框架就是php本身
Zend是車的引擎(發(fā)動機(jī))
Ext下面的各種組件就是車的輪子
Sapi可以看做是公路政恍,車可以跑在不同類型的公路上
而一次php程序的執(zhí)行就是汽車跑在公路上。
因此达传,我們需要:性能優(yōu)異的引擎+合適的車輪+正確的跑道

**Apache和php的關(guān)系**
Apache對于php的解析篙耗,就是通過眾多Module中的php Module來完成的。
[![](http://upload-images.jianshu.io/upload_images/2665779-25194743e2b0787f.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)](http://images.cnblogs.com/cnblogs_com/phphuaibei/201109/201109131718539690.png)
把php最終集成到Apache系統(tǒng)中宪赶,還需要對Apache進(jìn)行一些必要的設(shè)置宗弯。這里,我們就以php的mod_php5 SAPI運行模式為例進(jìn)行講解搂妻,至于SAPI這個概念后面我們還會詳細(xì)講解蒙保。
假定我們安裝的版本是Apache2 和 Php5,那么需要編輯Apache的主配置文件http.conf欲主,在其中加入下面的幾行內(nèi)容:
Unix/Linux環(huán)境下:

    LoadModule php5_module modules/mod_php5.so
    AddType application/x-httpd-php .php
注:其中 `modules/mod_php5.so` 是X系統(tǒng)環(huán)境下`mod_php5.so`文件的安裝位置邓厕。

Windows環(huán)境下:
    
    LoadModule php5_module d:/php/php5apache2.dll
    AddType application/x-httpd-php .php
注:其中 `d:/php/php5apache2.dll` 是在Windows環(huán)境下`php5apache2.dll`文件的安裝位置。
這兩項配置就是告訴Apache Server岛蚤,以后收到的Url用戶請求邑狸,凡是以php作為后綴,就需要調(diào)用php5_module模塊(mod_php5.so/ php5apache2.dll)進(jìn)行處理涤妒。

Apache的生命周期
[![](http://upload-images.jianshu.io/upload_images/2665779-f49a0d4035b35075.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)](http://images.cnblogs.com/cnblogs_com/phphuaibei/201109/201109131718581331.png)
Apach的請求處理流程
[![](http://upload-images.jianshu.io/upload_images/2665779-cccf97210396f091.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)](http://images.cnblogs.com/cnblogs_com/phphuaibei/201109/201109131719029809.png)
**Apache請求處理循環(huán)詳解**     
Apache請求處理循環(huán)的11個階段都做了哪些事情呢单雾?
1、Post-Read-Request階段
    在正常請求處理流程中,這是模塊可以插入鉤子的第一個階段硅堆。對于那些想很早進(jìn)入處理請求的模塊來說屿储,這個階段可以被利用。
    2渐逃、URI Translation階段     Apache在本階段的主要工作:將請求的URL映射到本地文件系統(tǒng)够掠。模塊可以在這階段插入鉤子,執(zhí)行自己的映射邏輯茄菊。mod_alias就是利用這個階段工作的疯潭。
    3、Header Parsing階段     Apache在本階段的主要工作:檢查請求的頭部面殖。由于模塊可以在請求處理流程的任何一個點上執(zhí)行檢查請求頭部的任務(wù)竖哩,因此這個鉤子很少被使用。mod_setenvif就是利用這個階段工作的脊僚。
    4相叁、Access Control階段     Apache在本階段的主要工作:根據(jù)配置文件檢查是否允許訪問請求的資源。Apache的標(biāo)準(zhǔn)邏輯實現(xiàn)了允許和拒絕指令辽幌。mod_authz_host就是利用這個階段工作的增淹。
    5、Authentication階段      Apache在本階段的主要工作:按照配置文件設(shè)定的策略對用戶進(jìn)行認(rèn)證乌企,并設(shè)定用戶名區(qū)域虑润。模塊可以在這階段插入鉤子,實現(xiàn)一個認(rèn)證方法逛犹。
    6端辱、Authorization階段     Apache在本階段的主要工作:根據(jù)配置文件檢查是否允許認(rèn)證過的用戶執(zhí)行請求的操作梁剔。模塊可以在這階段插入鉤子虽画,實現(xiàn)一個用戶權(quán)限管理的方法。
    7荣病、MIME Type Checking階段     Apache在本階段的主要工作:根據(jù)請求資源的MIME類型的相關(guān)規(guī)則码撰,判定將要使用的內(nèi)容處理函數(shù)。標(biāo)準(zhǔn)模塊mod_negotiation和mod_mime實現(xiàn)了這個鉤子个盆。
    8脖岛、FixUp階段     這是一個通用的階段,允許模塊在內(nèi)容生成器之前颊亮,運行任何必要的處理流程柴梆。和Post_Read_Request類似,這是一個能夠捕獲任何信息的鉤子终惑,也是最常使用的鉤子绍在。
    9、Response階段     Apache在本階段的主要工作:生成返回客戶端的內(nèi)容,負(fù)責(zé)給客戶端發(fā)送一個恰當(dāng)?shù)幕貜?fù)偿渡。這個階段是整個處理流程的核心部分臼寄。
    10、Logging階段     Apache在本階段的主要工作:在回復(fù)已經(jīng)發(fā)送給客戶端之后記錄事務(wù)溜宽。模塊可能修改或者替換Apache的標(biāo)準(zhǔn)日志記錄吉拳。
11、CleanUp階段     Apache在本階段的主要工作:清理本次請求事務(wù)處理完成之后遺留的環(huán)境适揉,比如文件留攒、目錄的處理或者Socket的關(guān)閉等等,這是Apache一次請求處理的最后一個階段嫉嘀。
**LAMP架構(gòu):**
[![wps_clip_image-24435](http://upload-images.jianshu.io/upload_images/2665779-5e5dbfe053192d21.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)](http://images.cnblogs.com/cnblogs_com/phphuaibei/201109/201109131719063585.png)
從下往上四層:
①liunx 屬于操作系統(tǒng)的底層
②apache服務(wù)器稼跳,屬于次服務(wù)器,溝通linux和PHP
③php:屬于服務(wù)端編程語言吃沪,通過php_module 模塊 和apache關(guān)聯(lián)
    ④mysql和其他web服務(wù):屬于應(yīng)用服務(wù)汤善,通過PHP的Extensions外 掛模塊和mysql關(guān)聯(lián)
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市票彪,隨后出現(xiàn)的幾起案子红淡,更是在濱河造成了極大的恐慌,老刑警劉巖降铸,帶你破解...
    沈念sama閱讀 222,865評論 6 518
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件在旱,死亡現(xiàn)場離奇詭異,居然都是意外死亡推掸,警方通過查閱死者的電腦和手機(jī)桶蝎,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,296評論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來谅畅,“玉大人登渣,你說我怎么就攤上這事≌毙海” “怎么了胜茧?”我有些...
    開封第一講書人閱讀 169,631評論 0 364
  • 文/不壞的土叔 我叫張陵,是天一觀的道長仇味。 經(jīng)常有香客問我呻顽,道長,這世上最難降的妖魔是什么丹墨? 我笑而不...
    開封第一講書人閱讀 60,199評論 1 300
  • 正文 為了忘掉前任廊遍,我火速辦了婚禮,結(jié)果婚禮上贩挣,老公的妹妹穿的比我還像新娘喉前。我一直安慰自己英染,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 69,196評論 6 398
  • 文/花漫 我一把揭開白布被饿。 她就那樣靜靜地躺著四康,像睡著了一般。 火紅的嫁衣襯著肌膚如雪狭握。 梳的紋絲不亂的頭發(fā)上闪金,一...
    開封第一講書人閱讀 52,793評論 1 314
  • 那天,我揣著相機(jī)與錄音论颅,去河邊找鬼哎垦。 笑死,一個胖子當(dāng)著我的面吹牛恃疯,可吹牛的內(nèi)容都是我干的漏设。 我是一名探鬼主播,決...
    沈念sama閱讀 41,221評論 3 423
  • 文/蒼蘭香墨 我猛地睜開眼今妄,長吁一口氣:“原來是場噩夢啊……” “哼郑口!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起盾鳞,我...
    開封第一講書人閱讀 40,174評論 0 277
  • 序言:老撾萬榮一對情侶失蹤犬性,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后腾仅,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體乒裆,經(jīng)...
    沈念sama閱讀 46,699評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,770評論 3 343
  • 正文 我和宋清朗相戀三年推励,在試婚紗的時候發(fā)現(xiàn)自己被綠了鹤耍。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,918評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡验辞,死狀恐怖稿黄,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情受神,我是刑警寧澤抛猖,帶...
    沈念sama閱讀 36,573評論 5 351
  • 正文 年R本政府宣布格侯,位于F島的核電站鼻听,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏联四。R本人自食惡果不足惜撑碴,卻給世界環(huán)境...
    茶點故事閱讀 42,255評論 3 336
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望朝墩。 院中可真熱鬧醉拓,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,749評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至排吴,卻和暖如春秆乳,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背钻哩。 一陣腳步聲響...
    開封第一講書人閱讀 33,862評論 1 274
  • 我被黑心中介騙來泰國打工屹堰, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人街氢。 一個月前我還...
    沈念sama閱讀 49,364評論 3 379
  • 正文 我出身青樓扯键,卻偏偏與公主長得像,于是被迫代替她去往敵國和親珊肃。 傳聞我的和親對象是個殘疾皇子荣刑,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,926評論 2 361

推薦閱讀更多精彩內(nèi)容

  • 轉(zhuǎn)自陳明乾的博客,可能有一定更新伦乔。 轉(zhuǎn)原文聲明:原創(chuàng)作品嘶摊,允許轉(zhuǎn)載,轉(zhuǎn)載時請務(wù)必以超鏈接形式標(biāo)明文章 原始出處 评矩、...
    C86guli閱讀 4,700評論 6 72
  • 那棵樹一直在那個角落站著 第一次見它是那個樣子 二十年后見它還是那個樣子 它都不老 一直不肯老 開花的時候 認(rèn)真開...
    王紅林閱讀 158評論 1 2
  • 第一不要“學(xué)舌” 這本小書的題目是《怎樣練習(xí)寫作》叶堆。所謂“寫作”,范圍應(yīng)當(dāng)很廣斥杜,凡是敘事虱颗、抒情、議論蔗喂,乃至日常應(yīng)用...
    駱炎堂ETAN閱讀 583評論 0 9
  • 關(guān)于“成功學(xué)”忘渔,很多人是聽過的,甚至有些人是參與過的缰儿。幾年前畦粮,在成功學(xué)盛行的時候,幾乎到處都是成功學(xué)大師的宣傳乖阵,到...
    銳社長閱讀 1,519評論 0 3
  • 28XXXX-0-小勖 我知道老大是萬能的钩蚊,所以斗膽問了2個問題贡翘,望看見后能回復(fù)! 1砰逻、準(zhǔn)備辭職鸣驱,沒有目標(biāo)的裸辭,...
    雙魚若曦閱讀 919評論 0 1