web伺服器上PHP的執行模式
CGI通用閘道器介面(Common Gateway Interface)
CGI即通用閘道器介面(Common Gateway Interface),它是一段程式, 通俗的講CGI就象是一座橋,把網頁和WEB伺服器中的執行程式連線起來,它把HTML接收的指令傳遞給伺服器的執行程式,再把伺服器執行程式的結果返還給HTML頁。CGI 的跨平臺效能極佳,幾乎可以在任何作業系統上實現。 CGI已經是比較老的模式了,這幾年都很少用了。
每有一個使用者請求,都會先要建立cgi的子程序,然後處理請求,處理完後結束這個子程序,這就是fork-and-execute模式。 當用戶請求數量非常多時,會大量擠佔系統的資源如記憶體,CPU時間等,造成效能低下。所以用cgi方式的伺服器有多少連線請求就會有多少cgi子程序,子程序反覆載入是cgi效能低下的主要原因。
如果不想把 PHP 嵌入到伺服器端軟體(如 Apache)作為一個模組安裝的話,可以選擇以 CGI 的模式安裝。或者把 PHP 用於不同的 CGI 封裝以便為程式碼建立安全的 chroot 和 setuid 環境。這樣每個客戶機請求一個php檔案,Web伺服器就呼叫php.exe(win下是php.exe,linux是php)去解釋這個檔案,然後再把解釋的結果以網頁的形式返回給客戶機。 這種安裝方式通常會把 PHP 的可執行檔案安裝到 web 伺服器的 cgi-bin 目錄。CERT 建議書 CA-96.11 建議不要把任何的直譯器放到 cgi-bin 目錄。
這種方式的好處是把web server和具體的程式處理獨立開來,結構清晰,可控性強,同時缺點就是如果在高訪問需求的情況下,cgi的程序fork就會成為很大的伺服器負擔,想 象一下數百個併發請求導致伺服器fork出數百個程序就明白了。這也是為什麼cgi一直揹負效能低下,高資源消耗的惡名的原因。
fast-cgi (常駐 (long-live) 型的 CGI)
cgi模式的變種,也就是(cgi是普通人,fast-cgi是特種兵)
FastCGI 像是一個常駐 (long-live) 型的 CGI,它可以一直執行著,只要啟用後,不會每次都要花費時間去 fork 一次 (這是 CGI 最為人詬病的 fork-and-execute 模式)。
FastCGI的工作原理是:
- Web Server啟動時載入FastCGI程序管理器【PHP的FastCGI程序管理器是PHP-FPM(php-FastCGI Process Manager)】(IIS ISAPI或Apache Module);
- FastCGI程序管理器自身初始化,啟動多個CGI直譯器程序 (在工作管理員中可見多個php-cgi.exe)並等待來自Web Server的連線。
- 當客戶端請求到達Web Server時,FastCGI程序管理器選擇並連線到一個CGI直譯器。Web server將CGI環境變數和標準輸入傳送到FastCGI子程序php-cgi。
- FastCGI子程序完成處理後將標準輸出和錯誤資訊從同一連線返回Web Server。當FastCGI子程序關閉連線時,請求便告處理完成。FastCGI子程序接著等待並處理來自FastCGI程序管理器(執行在 WebServer中)的下一個連線。在正常的CGI模式中,php-cgi.exe在此便退出了。
在CGI模式中,你可以想象 CGI通常有多慢。每一個Web請求PHP都必須重新解析php.ini、重新載入全部dll擴充套件並重初始化全部資料結構。使用FastCGI,所有這些都只在程序啟動時發生一次。一個額外的好處是,持續資料庫連線(Persistent database connection)可以工作
cli 命令列執行(Command Line Interface)
在終端敲php script.php等命令列
web模組模式(apache等web伺服器執行的模組模式)
模組模式是以mod_php5模組的形式整合,此時mod_php5模組的作用是接收Apache傳遞過來的PHP檔案請求,並處理這些請求,然後將處理後的結果返回給Apache,是apache在CGI的基礎上進行的一種擴充套件,加快PHP的執行效率。如果我們在Apache啟動前在其配置檔案中配置好了PHP模組(mod_php5), PHP模組通過註冊apache2的ap_hook_post_config掛鉤,在Apache啟動的時候啟動此模組以接受PHP檔案的請求。
除了這種啟動時的載入方式,Apache的模組可以在執行的時候動態裝載,這意味著對伺服器可以進行功能擴充套件而不需要重新對原始碼進行編譯,甚至根本不需要停止伺服器。我們所需要做的僅僅是給伺服器傳送訊號HUP或者AP_SIG_GRACEFUL通知伺服器重新載入模組。但是在動態載入之前,我們需要將模組編譯成為動態連結庫。此時的動態載入就是載入動態連結庫。 Apache中對動態連結庫的處理是通過模組mod_so來完成的,因此mod_so模組不能被動態載入,它只能被靜態編譯進Apache的核心。這意味著它是隨著Apache一起啟動的。
Apache是如何載入模組的呢?首先我們需要在Apache的配置檔案httpd.conf中新增一行:
LoadModule php5_module modules/mod_php5.so
這裡我們使用了LoadModule命令,該命令的第一個引數是模組的名稱,名稱可以在模組實現的原始碼中找到。第二個選項是該模組所處的路徑。如果需要在伺服器執行時載入模組,可以通過傳送訊號HUP或者AP_SIG_GRACEFUL給伺服器,一旦接受到該訊號,Apache將重新裝載模組,而不需要重新啟動伺服器。