背景
PHP最常用的方式是以模塊的方式(mod_php)運(yùn)行在A(yíng)pache中椿息,也是Apache運(yùn)行PHP的默認(rèn)方式歹袁;但在Nginx中,Nginx又使用的是PHP-FPM寝优,但是PHP-FPM到底是個(gè)什么東東条舔?跟php有什么關(guān)系?今天我們一起來(lái)探究一番
PHP處理器(PHP handlers)
首先需要記住的是乏矾,任何一種Web服務(wù)器(Apache孟抗、Nginx等)都是被設(shè)計(jì)成向用戶(hù)發(fā)送html、圖片等靜態(tài)資源的钻心,Web服務(wù)器自身并不能解釋任何動(dòng)態(tài)腳本(PHP凄硼、Python等)
PHP處理器就是用來(lái)解釋W(xué)eb應(yīng)用中的PHP代碼,并將它解釋為HTML或其他靜態(tài)資源捷沸,然后將解析的結(jié)果傳給Web服務(wù)器摊沉,最后再由Web服務(wù)器發(fā)送給用戶(hù)
大多數(shù)的Web服務(wù)器都不能解析PHP代碼,因此它需要一個(gè)能解析PHP代碼的程序痒给,這就是PHP處理器
現(xiàn)在我們知道了说墨,Apache與Nginx都需要PHP處理器來(lái)處理php代碼骏全,那么怎么連接上服務(wù)器與php處理器呢?也就是說(shuō)服務(wù)器與php處理器如何通信尼斧?
答案是通過(guò)SAPI(Server Application Programming Interface 服務(wù)器端應(yīng)用編程端口)姜贡,簡(jiǎn)單來(lái)說(shuō),SAPI指的是PHP具體應(yīng)用的編程接口棺棵, 就像PC一樣鲁豪,無(wú)論安裝哪些操作系統(tǒng),只要滿(mǎn)足了PC的接口規(guī)范都可以在PC上正常運(yùn)行律秃, PHP腳本要執(zhí)行有很多種方式爬橡,通過(guò)Web服務(wù)器,或者直接在命令行下棒动,也可以嵌入在其他程序中糙申,有興趣大家可以研究PHP內(nèi)核
我們這里繼續(xù)討論P(yáng)HP最常用的SAPI提供的2種連接方法:mod_php和mod_fastcgi
mod_php模式
咱們回顧一下,Apache是怎么能夠識(shí)別php代碼的船惨?是不是Apache的配置文件httpd.conf中加上或者修改這樣幾句:
1234567
//添加LoadModule php5_module modules/libphp5.soAddType application/x-httpd-php .php//修改<IfModule dir_module> DirectoryIndex index.php index.html index.htm index.html</IfModule>
也即php作為Apache的一個(gè)子模塊來(lái)運(yùn)行柜裸,當(dāng)通過(guò)web訪(fǎng)問(wèn)php文件時(shí),Apache就會(huì)調(diào)用php5_module來(lái)解析php代碼
配置加載mod_php模塊后粱锐,php便是Apahce進(jìn)程本身一部分疙挺,每個(gè)新的Apache子進(jìn)程都會(huì)加載此模塊
mod_fastcgi模式
我們先看PHP-FPM官網(wǎng)的說(shuō)明
PHP-FPM - A simple and robust FastCGI Process Manager for PHPPHP-FPM (FastCGI Process Manager) is an alternative PHP FastCGI implementation with some additional features useful for sites of any size, especially busier sites.
PHP-FPM是一個(gè)PHP的FastCGI進(jìn)程管理器,解釋的非常簡(jiǎn)單怜浅。這說(shuō)明PHP-FPM是輔助mod_fastcgi模式進(jìn)行工作的铐然,然而FastCGI又是個(gè)什么東西?管理著什么進(jìn)程恶座?
什么是CGI搀暑?
CGI(Common Gateway Interface) 是WWW技術(shù)中最重要的技術(shù)之一,有著不可替代的重要地位
CGI是外部應(yīng)用程序(CGI程序)與Web服務(wù)器之間的接口標(biāo)準(zhǔn)跨琳,是在CGI程序和Web服務(wù)器之間傳遞信息的規(guī)程
CGI規(guī)范允許Web服務(wù)器執(zhí)行外部程序自点,并將它們的輸出發(fā)送給Web瀏覽器,CGI將Web的一組簡(jiǎn)單的靜態(tài)超媒體文檔變成一個(gè)完整的新的交互式媒體
說(shuō)白了脉让,CGI是一種外部應(yīng)用程序(CGI程序)與Web服務(wù)器的協(xié)議桂敛,CGI是為了保證Server傳遞過(guò)來(lái)的數(shù)據(jù)是標(biāo)準(zhǔn)格式
什么是FastCGI?
FastCGI像是一個(gè)常駐(long-live)型的CGI溅潜,它可以一直執(zhí)行著术唬,只要激活后,不會(huì)每次都要花費(fèi)時(shí)間去fork一次(這是CGI最為人詬病的fork-and-execute 模式)伟恶。它還支持分布式的運(yùn)算, 即 FastCGI 程序可以在網(wǎng)站服務(wù)器以外的主機(jī)上執(zhí)行并且接受來(lái)自其它網(wǎng)站服務(wù)器來(lái)的請(qǐng)求
FastCGI是語(yǔ)言無(wú)關(guān)的碴开、可伸縮架構(gòu)的CGI開(kāi)放擴(kuò)展毅该,其主要行為是將CGI解釋器進(jìn)程保持在內(nèi)存中并因此獲得較高的性能博秫。眾所周知潦牛,CGI解釋器的反復(fù)加載是CGI性能低下的主要原因,如果CGI解釋器保持在內(nèi)存中并接受FastCGI進(jìn)程管理器調(diào)度挡育,則可以提供良好的性能巴碗、伸縮性、Fail- Over特性等等
一般情況下即寒,F(xiàn)astCGI的整個(gè)工作流程是這樣的:
Web Server啟動(dòng)時(shí)載入FastCGI進(jìn)程管理器(IIS ISAPI或Apache Module)
FastCGI進(jìn)程管理器自身初始化橡淆,啟動(dòng)多個(gè)CGI解釋器進(jìn)程(可見(jiàn)多個(gè)php-cgi)并等待WebServer的連接
當(dāng)客戶(hù)端請(qǐng)求到達(dá)Web Server時(shí),F(xiàn)astCGI進(jìn)程管理器選擇并連接到一個(gè)CGI解釋器母赵。 Web server將CGI環(huán)境變量和標(biāo)準(zhǔn)輸入發(fā)送到FastCGI子進(jìn)程php-cgi
FastCGI子進(jìn)程完成處理后將標(biāo)準(zhǔn)輸出和錯(cuò)誤信息從同一連接返回Web Server逸爵。當(dāng)FastCGI子進(jìn)程關(guān)閉連接時(shí),請(qǐng)求便告處理完成凹嘲,F(xiàn)astCGI子進(jìn)程接著等待并處理來(lái)自FastCGI進(jìn)程管理器(運(yùn)行在Web Server中)的下一個(gè)連接师倔,在CGI模式中,php-cgi在此便已經(jīng)退出
也就是說(shuō)FastCGI是CGI的升級(jí)版周蹭,一種語(yǔ)言無(wú)關(guān)的協(xié)議趋艘,用來(lái)溝通程序(如PHP, Python, Java)和Web服務(wù)器(Apache2, Nginx), 理論上任何語(yǔ)言編寫(xiě)的程序都可以通過(guò)FastCGI來(lái)提供Web服務(wù)
FastCGI的特點(diǎn)是會(huì)在一個(gè)進(jìn)程中依次完成多個(gè)請(qǐng)求,以達(dá)到提高效率的目的凶朗,大多數(shù)FastCGI實(shí)現(xiàn)都會(huì)維護(hù)一個(gè)進(jìn)程池
通俗解釋?zhuān)篎astCGI事先就需要啟動(dòng)瓷胧,而且可以啟動(dòng)多個(gè)CGI模塊,在那里一直運(yùn)行等著web發(fā)請(qǐng)求棚愤,然后再給php解析運(yùn)算搓萧,完成后生成html返回給web后,但是完成后它不會(huì)退出宛畦,而是繼續(xù)等著下一個(gè)web請(qǐng)求
PHP-FPM
PHP-FPM就是針對(duì)于PHP的FastCGI的一種實(shí)現(xiàn)矛绘,他負(fù)責(zé)管理一個(gè)進(jìn)程池,來(lái)處理來(lái)自Web服務(wù)器的請(qǐng)求
但是PHP-FPM僅僅是個(gè)“PHP FastCGI 進(jìn)程管理器”, 它仍會(huì)調(diào)用PHP解釋器本身來(lái)處理請(qǐng)求刃永,PHP解釋器(在Windows下)就是php-cgi.exe