php-fpm的配置和優化

無敵小籠包發表於2019-02-16

centos等linux平臺

/usr/local/php/php
/usr/local/php/etc/php.ini
/usr/local/php/sbin/php-fpm
/usr/local/php/etc/php-fpm.conf

mac平臺

/usr/bin/php
/etc/php.ini
/usr/bin/php-fpm
/etc/php-fpm.conf

由於我開發以Mac為主,所以就用Mac的環境配置來學習。

php-fpm配置詳解

這是搜尋的一份還算算比較詳細的php-fpm.conf配置詳解,我會針對性的修改下,當然php手冊上也有詳細的講解:http://php.net/manual/zh/inst…

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 . 預設值: 空.

當然還有一些無關緊要的設定,用到了再說吧。

一些重要的設定

php-fpm程式分配

fasgcgi模式下,php會啟動多個php-fpm程式,來接收nginx發來的請求,那是不是程式越多,速度就越快呢?這可不一定!得根據我們的機器配置和業務量來決定。

我們先來看下,設定程式的配置在哪裡?

pm = static | dynamic | ondemand
pm可以設定成這樣3種,我們用的最多的就上前面2種。

pm = static 模式

pm = static 表示我們建立的php-fpm子程式數量是固定的,那麼就只有pm.max_children = 50這個引數生效。你啟動php-fpm的時候就會一起全部啟動51(1個主+50個子)個程式,頗為壯觀。

pm = dynamic 模式

pm = dynamic模式,表示啟動程式是動態分配的,隨著請求量動態變化的。他由 pm.max_children,pm.start_servers,pm.min_spare_servers,pm.max_spare_servers 這幾個引數共同決定。

上面已經講過,這裡再重申一下吧:

pm.max_children = 50 是最大可建立的子程式的數量。必須設定。這裡表示最多隻能50個子程式。

pm.start_servers = 20 隨著php-fpm一起啟動時建立的子程式數目。預設值:min_spare_servers + (max_spare_servers - min_spare_servers) / 2。這裡表示,一起啟動會有20個子程式。

pm.min_spare_servers = 10 
設定伺服器空閒時最小php-fpm程式數量。必須設定。如果空閒的時候,會檢查如果少於10個,就會啟動幾個來補上。

pm.max_spare_servers = 30 

設定伺服器空閒時最大php-fpm程式數量。必須設定。如果空閒時,會檢查程式數,多於30個了,就會關閉幾個,達到30個的狀態。

到底選擇static還數dynamic?

很多人恐懼症來襲,不知道選什麼好?

一般原則是:動態適合小記憶體機器,靈活分配程式,省記憶體。靜態適用於大記憶體機器,動態建立回收程式對伺服器資源也是一種消耗。

如果你的記憶體很大,有8-20G,按照一個php-fpm程式20M算,100個就2G記憶體了,那就可以開啟static模式。如果你的記憶體很小,比如才256M,那就要小心設定了,因為你的機器裡面的其他的程式也算需要佔用記憶體的,所以設定成dynamic是最好的,比如:pm.max_chindren = 8, 佔用記憶體160M左右,而且可以隨時變化,對於一半訪問量的網站足夠了。

慢日誌查詢

我們有時候會經常飽受500,502問題困擾。當nginx收到如上錯誤碼時,可以確定後端php-fpm解析php出了某種問題,比如,執行錯誤,執行超時。

這個時候,我們是可以開啟慢日誌功能的。

slowlog = /usr/local/var/log/php-fpm.log.slow
request_slowlog_timeout = 15s

當一個請求該設定的超時時間15秒後,就會將對應的PHP呼叫堆疊資訊完整寫入到慢日誌中。
php-fpm慢日誌會記錄下程式號,指令碼名稱,具體哪個檔案哪行程式碼的哪個函式執行時間過長:

[21-Nov-2013 14:30:38] [pool www] pid 11877
script_filename = /usr/local/lnmp/nginx/html/www.quancha.cn/www/fyzb.php
[0xb70fb88c] file_get_contents() /usr/local/lnmp/nginx/html/www.quancha.cn/www/fyzb.php:2

通過日誌,我們就可以知道第2行的file_get_contents 函式有點問題,這樣我們就能追蹤問題了。

相關文章