1. 程式人生 > >php中fastcgi和php-fpm是什麼

php中fastcgi和php-fpm是什麼

有一些工具一直在用,卻從來也不知道這些東西是什麼。

現在想想還是梳理一下比較好。所以看到我覺得不錯的定義就抄過來了。

首先,CGI是幹嘛的?CGI是為了保證web server傳遞過來的資料是標準格式的,方便CGI程式的編寫者。

web server(比如說nginx)只是內容的分發者。比如,如果請求/index.html,那麼web server會去檔案系統中找到這個檔案,傳送給瀏覽器,這裡分發的是靜態資料。好了,如果現在請求的是/index.php,根據配置檔案,nginx知道這個不是靜態檔案,需要去找PHP解析器來處理,那麼他會把這個請求簡單處理後交給PHP解析器。Nginx會傳哪些資料給PHP解析器呢?url要有吧,查詢字串也得有吧,POST資料也要有,HTTP header不能少吧,好的,CGI就是規定要傳哪些資料、以什麼樣的格式傳遞給後方處理這個請求的協議。仔細想想,你在PHP程式碼中使用的使用者從哪裡來的。

當web server收到/index.php這個請求後,會啟動對應的CGI程式,這裡就是PHP的解析器。接下來PHP解析器會解析php.ini檔案,初始化執行環境,然後處理請求,再以規定CGI規定的格式返回處理後的結果,退出程序。web server再把結果返回給瀏覽器。

好了,CGI是個協議,跟程序什麼的沒關係。那fastcgi又是什麼呢?Fastcgi是用來提高CGI程式效能的。

提高效能,那麼CGI程式的效能問題在哪呢?“PHP解析器會解析php.ini檔案,初始化執行環境”,就是這裡了。標準的CGI對每個請求都會執行這些步驟(不閒累啊!啟動程序很累的說!),所以處理每個時間的時間會比較長。這明顯不合理嘛!那麼Fastcgi是怎麼做的呢?首先,Fastcgi會先啟一個master,解析配置檔案,初始化執行環境,然後再啟動多個worker。當請求過來時,master會傳遞給一個worker,然後立即可以接受下一個請求。這樣就避免了重複的勞動,效率自然是高。而且當worker不夠用時,master可以根據配置預先啟動幾個worker等著;當然空閒worker太多時,也會停掉一些,這樣就提高了效能,也節約了資源。這就是fastcgi的對程序的管理。

那PHP-FPM又是什麼呢?是一個實現了Fastcgi的程式,被PHP官方收了。

大家都知道,PHP的直譯器是php-cgi。php-cgi只是個CGI程式,他自己本身只能解析請求,返回結果,不會程序管理(皇上,臣妾真的做不到啊!)所以就出現了一些能夠排程php-cgi程序的程式,比如說由lighthttpd分離出來的spawn-fcgi。好了PHP-FPM也是這麼個東東,在長時間的發展後,逐漸得到了大家的認可(要知道,前幾年大家可是抱怨PHP-FPM穩定性太差的),也越來越流行。

fastcgi是一個協議,php-fpm實現了這個協議

對。

有的說,php-fpm是fastcgi程序的管理器,用來管理fastcgi程序的

對。php-fpm的管理物件是php-cgi。但不能說php-fpm是fastcgi程序的管理器,因為前面說了fastcgi是個協議,似乎沒有這麼個程序存在,就算存在php-fpm也管理不了他(至少目前是)。 有的說,php-fpm是php核心的一個補丁

以前是對的。因為最開始的時候php-fpm沒有包含在PHP核心裡面,要使用這個功能,需要找到與原始碼版本相同的php-fpm對核心打補丁,然後再編譯。後來PHP核心集成了PHP-FPM之後就方便多了,使用–enalbe-fpm這個編譯引數即可。

有的說,修改了php.ini配置檔案後,沒辦法平滑重啟,所以就誕生了php-fpm

是的,修改php.ini之後,php-cgi程序的確是沒辦法平滑重啟的。php-fpm對此的處理機制是新的worker用新的配置,已經存在的worker處理完手上的活就可以歇著了,通過這種機制來平滑過度。

還有的說PHP-CGI是PHP自帶的FastCGI管理器,那這樣的話幹嗎又弄個php-fpm出

不對。php-cgi只是解釋PHP指令碼的程式而已。

php-fpm的一些重要配置

pid = /usr/local/var/run/php-fpm.pid
#pid設定,一定要開啟,上面是Mac平臺的。預設在php安裝目錄中的var/run/php-fpm.pid。比如centos的在: /usr/local/php/var/run/php-fpm.pid
error_log  = /usr/local/var/log/php-fpm.log
#錯誤日誌,上面是Mac平臺的,預設在php安裝目錄中的var/log/php-fpm.log,比如centos的在: /usr/local/php/var/log/php-fpm.log
log_level = notice
#錯誤級別. 上面的php-fpm.log紀錄的登記。可用級別為: alert(必須立即處理), error(錯誤情況), warning(警告情況), notice(一般重要資訊), debug(除錯資訊). 預設: notice.
emergency_restart_threshold = 60
emergency_restart_interval = 60s
#表示在emergency_restart_interval所設值內出現SIGSEGV或者SIGBUS錯誤的php-cgi程序數如果超過 emergency_restart_threshold個,php-fpm就會優雅重啟。這兩個選項一般保持預設值。0 表示 '關閉該功能'. 預設值: 0 (關閉).
process_control_timeout = 0
#設定子程序接受主程序複用訊號的超時時間. 可用單位: s(秒), m(分), h(小時), 或者 d(天) 預設單位: s(秒). 預設值: 0.
daemonize = yes
#後臺執行fpm,預設值為yes,如果為了除錯可以改為no。在FPM中,可以使用不同的設定來執行多個程序池。 這些設定可以針對每個程序池單獨設定。
listen = 127.0.0.1:9000
#fpm監聽埠,即nginx中php處理的地址,一般預設值即可。可用格式為: 'ip:port', 'port', '/path/to/unix/socket'. 每個程序池都需要設定。如果nginx和php在不同的機器上,分散式處理,就設定ip這裡就可以了。
listen.backlog = -1
#backlog數,設定 listen 的半連線佇列長度,-1表示無限制,由作業系統決定,此行註釋掉就行。backlog含義參考:http://www.3gyou.cc/?p=41
listen.allowed_clients = 127.0.0.1
#允許訪問FastCGI程序的IP白名單,設定any為不限制IP,如果要設定其他主機的nginx也能訪問這臺FPM程序,listen處要設定成本地可被訪問的IP。預設值是any。每個地址是用逗號分隔. 如果沒有設定或者為空,則允許任何伺服器請求連線。
listen.owner = www
listen.group = www
listen.mode = 0666
#unix socket設定選項,如果使用tcp方式訪問,這裡註釋即可。
user = www
group = www
#啟動程序的使用者和使用者組,FPM 程序執行的Unix使用者, 必須要設定。使用者組,如果沒有設定,則預設使用者的組被使用。
pm = dynamic 
#php-fpm程序啟動模式,pm可以設定為static和dynamic和ondemand
#如果選擇static,則程序數就數固定的,由pm.max_children指定固定的子程序數。
#如果選擇dynamic,則程序數是動態變化的,由以下引數決定:
pm.max_children = 50 #子程序最大數
pm.start_servers = 2 #啟動時的程序數,預設值為: min_spare_servers + (max_spare_servers - min_spare_servers) / 2
pm.min_spare_servers = 1 #保證空閒程序數最小值,如果空閒程序小於此值,則建立新的子程序
pm.max_spare_servers = 3 #,保證空閒程序數最大值,如果空閒程序大於此值,此進行清理
pm.max_requests = 500
#設定每個子程序重生之前服務的請求數. 對於可能存在記憶體洩漏的第三方模組來說是非常有用的. 如果設定為 '0' 則一直接受請求. 等同於 PHP_FCGI_MAX_REQUESTS 環境變數. 預設值: 0.
pm.status_path = /status
#FPM狀態頁面的網址. 如果沒有設定, 則無法訪問狀態頁面. 預設值: none. munin監控會使用到
ping.path = /ping
#FPM監控頁面的ping網址. 如果沒有設定, 則無法訪問ping頁面. 該頁面用於外部檢測FPM是否存活並且可以響應請求. 請注意必須以斜線開頭 (/)。
ping.response = pong
#用於定義ping請求的返回相應. 返回為 HTTP 200 的 text/plain 格式文字. 預設值: pong.
access.log = log/$pool.access.log
#每一個請求的訪問日誌,預設是關閉的。
access.format = "%R - %u %t \"%m %r%Q%q\" %s %f %{mili}d %{kilo}M %C%%"
#設定訪問日誌的格式。
slowlog = log/$pool.log.slow
#慢請求的記錄日誌,配合request_slowlog_timeout使用,預設關閉
request_slowlog_timeout = 10s
#當一個請求該設定的超時時間後,就會將對應的PHP呼叫堆疊資訊完整寫入到慢日誌中. 設定為 '0' 表示 'Off'
request_terminate_timeout = 0
#設定單個請求的超時中止時間. 該選項可能會對php.ini設定中的'max_execution_time'因為某些特殊原因沒有中止執行的指令碼有用. 設定為 '0' 表示 'Off'.當經常出現502錯誤時可以嘗試更改此選項。
rlimit_files = 1024
#設定檔案開啟描述符的rlimit限制. 預設值: 系統定義值預設可開啟控制代碼是1024,可使用 ulimit -n檢視,ulimit -n 2048修改。
rlimit_core = 0
#設定核心rlimit最大限制值. 可用值: 'unlimited' 、0或者正整數. 預設值: 系統定義值.
chroot =
#啟動時的Chroot目錄. 所定義的目錄需要是絕對路徑. 如果沒有設定, 則chroot不被使用.
chdir =
#設定啟動目錄,啟動時會自動Chdir到該目錄. 所定義的目錄需要是絕對路徑. 預設值: 當前目錄,或者/目錄(chroot時)
catch_workers_output = yes
#重定向執行過程中的stdout和stderr到主要的錯誤日誌檔案中. 如果沒有設定, stdout 和 stderr 將會根據FastCGI的規則被重定向到 /dev/null . 預設值: 空.