PHP 程式管理器 PHP-fpm 配置說明

奕鵬發表於2020-05-22

php-fpm是什麼

php-fpm是PHP的一個程式管理器。php下面的眾多work程式皆有php-fpm程式管理器管理。具體可參考:www.qqdeveloper.com/2019/10/06/CGI-...

php-fpm的工作原理

php-fpm全名是PHP FastCGI程式管理器。php-fpm啟動後會先讀php.ini,然後再讀相應的conf配置檔案,conf配置可以覆蓋php.ini的配置。
啟動php-fpm之後,會建立一個master程式,監聽9000埠(可配置),master程式又會根據fpm.conf/www.conf去建立若干子程式,子程式用於處理實際的業務。
當有客戶端(比如nginx)來連線9000埠時,空閒子程式會自己去accept,如果子程式全部處於忙碌狀態,新進的待accept的連線會被master放進佇列裡,等待fpm子程式空閒;這個存放待accept的半連線的佇列有多長,由listen.backlog 配置。

如何檢視php-fpm程式與子程式

檢視php-fpm相關的所有程式。

這裡pool www皆是php-fpm的子程式,也就是我們常說的work程式。

檢視php-fpm下面的子程式

通過上面的命令,其實我們能夠看出php-fpm相關的程式了,如果我們需要更加直觀的檢視php-fpm的master程式和work程式,可以通過下面的方式程式檢視。
這裡的5370則是php-fpm的master程式號。通過上面的命令已經很能直觀的得出。

通過上面的命令,可以看出php-fpm作為master程式,下面有15個子程式。這裡的子程式數都是可以程式自定義配置。通過如下幾個引數程式配置:

pm = dynamic # 動態建立子程式
pm.max_children = 20 # 最大子程式數
pm.start_servers = 15 # 初始化php-fpm程式時,預設的子程式數

php-fpm引數配置說明

php-fpm全域性配置引數

#php-fpm的執行許可權。
#以什麼使用者什麼組的許可權來執行池fpm。
user = www
group = www

#php-fpm的執行方式,可以使埠,也可以使socke檔案。
#如果是埠則是走tcp,如果是socket則直接讀socket檔案,這樣效能相對更好。
listen = 127.0.0.1:9000 

#擁有socket許可權的使用者,需要和上面的user、group配置相結合。
#如果採用的埠的方式,則不配置。
listen.owner = www
listen.group = www
listen.mode = 0660

#這是php-fpm埠連線的地址。多個用","隔開。預設任意地址都可以連線。
#例如Nginx和php-fpm不在同一臺伺服器上,這裡的值就是Nginx服務的ip地址。
#當Nginx和php-fpm配置在同一臺伺服器上,則直接寫127.0.0.1即可。
listen.allowed_clients = 127.0.0.1 

#pid程式檔案存放的位置,當我們啟用一個php服務,
#則會自動建立一個該pid檔案,其實我們可以直接把該檔案理解理解php-fpm的程式號檔案,
#兩則是等價的。預設為none。
pid = /opt/remi/php72/root/var/run/php-fpm/php-fpm.pid 

#錯誤日誌位置,預設:安裝路徑 #INSTALL_PREFIX#/log/php-fpm.log。
#如果設定為syslog,log就會傳送給syslogd服務而不會寫進檔案裡。
error_log = /opt/remi/php72/root/var/log/php-fpm/error.log 

#PHP限制的副檔名
security.limit_extensions = .php .php3 .php4 .php5 .php7

#系統日誌標示,如果跑了多個fpm程式,需要用這個來區分日誌是誰的。
syslog.ident = php-fpm 

#日記登記,可選:alert, error, warning, notice, debug。
log_level = notice 

#緊急重啟閾值,需要與下面emergency_restart_interval引數一起配置。
emergency_restart_threshold = 60 

# 緊急重啟閾值的時間範圍。在此引數設定的時間內,
# 出現SIGSEGV或SIGBUS的子程式數超過emergency_restart_threshold引數設定的值。
# 那麼fpm就會優雅的重啟,值是0表示off這個功能,可用的單位有:s秒,m分,h時,d天。
emergency_restart_interval = 60s 

#設定子程式接受主程式複用訊號的超時時間。
process_control_timeout = 0 

#當動態管理子程式時,fpm最多能fork多少個程式,0表示無限制,
# 這是所有程式池能啟動子程式的總和,謹慎使用。
process.max = 128 

#設定子程式的優先順序,在master程式以root使用者啟動時有效;
#如果沒有設定,子程式會繼承master程式的優先順序,值範圍-19(最高)到20(最低),預設不設定。
process.priority = -19 

#設定成no用於除錯bug,預設為yes。
daemonize = yes 

#master程式最多能開啟的檔案數量。預設採用系統設定的值。
rlimit_files = 1024 

#master程式核心rlimit限制值;可選unlimited或>=0的整數,預設為系統的值。
rlimit_core = 0

#事件處理機制,預設自動檢測,可選值:select,poll,
#epoll(linux>=2.5.44),kqueue,/dev/poll,port
events.mechanism = epoll 

#fpm想系統傳送狀態的頻率。單位有s,m,h。
#前提是fpm被設定會系統服務。
systemd_interval = 10s 

php-fpm的程式程式池配置

#php-fpm的佇列長度。
listen.backlog = 65535 

#php程式池許可權,同樣要master程式是root使用者才有效,
#和上面的全域性設定一樣,不設定的話會繼承master程式的優先順序。
process.priority = -19 

#子程式管理方式
#static(靜態配置,在啟動php-fpm時根據該值建立固定的子程式數量);
#dynamic(動態配置,在啟動php-fpm時根據pm.start_servers的值初始化對應的子程式數,至少一個子程式);
#ondemand(按需配置,在啟動php-fpm時不建立子程式,而是根據請求動態fork子程式);
pm = dynamic 

#最大子程式數量
pm.max_children = 5 

#初始化子程式數量,與上面的pm = dynamic配置使用。
pm.start_servers = 2 

#伺服器閒置時最少保持2個子程式,不夠這個數就會建立,只適用動態dynamic管理方式
pm.min_spare_servers = 2 

#伺服器閒置時最多要有幾個,多了會kill,只適用動態dynamic管理方式
pm.max_spare_servers = 3 

#子程式閒置時間,也就是說子程式沒有可處理的任務時,在該之間使就會被killed。
pm.process_idle_timeout = 10s

#每個子程式最大的處理請求數量。在一定程度上可以防止記憶體洩漏。
pm.max_requests = 500 

#php-fpm狀態監控的uri
pm.status_path string

#php-fpm監控頁面的 ping 網址。
#如果沒有設定,則無法訪問 ping 頁面。
#該頁面用於外部檢測php-fpm是否存活並且可以響應請求。請注意必須以斜線開頭(/)。
ping.path string

#用於定義ping請求的返回響應。返回為 HTTP 200 的 text/plain 格式文字。預設值:pong。
ping.response string

#設定worker的nice(2)優先順序(如果設定了的話)。
#該值從 -19(最高優先順序) 到 20(更低優先順序)。 
#預設值:不設定
process.priority int

#檢測路徑時使用的字首
prefix string

#訪問檔案日誌,沒啥用處,比如yii2每次都記錄訪問index.php,只是記錄真實的PHP檔案。
access.log = var/log/$pool.access.log 

#php的慢日誌
slowlog = var/log/$pool.log.slow 

#慢日誌時間閾值
request_slowlog_timeout = 2s 

#單個請求的超時時間,當php.ini設定的最大執行時間未生效,則交由它來處理。
request_terminate_timeout = 3s 

#最大開啟控制程式碼數,預設為系統值。
rlimit_files = 1024 

#最多的核心使用數,預設為系統分配。
rlimit_core = 0 

部分配置演示

php-fpm的backlog大小設定

php-fpm的backlog大小設定與php-fpm的處理能力有關,而不是越大越好。

當該值設定過大,導致php-fpm處理不過來,nginx那邊等待超時,斷開連線,報504 gateway timeout錯。同時php-fpm處理完準備write 資料給nginx時,發現TCP連線斷開了,報“Broken pipe”。

當該值設定過小,nginx之類的client請求,根本進入不了php-fpm的accept queue,報“502 Bad Gateway”錯。所以,這還得去根據php-fpm的QPS來決定backlog的大小。計算方式最好為QPS=backlog。
具體可以參考:www.jianshu.com/p/3ecc99ebf566

php-fpm啟動模式

php-fpm以socket啟動或者埠啟動,這兩種的方式根據實際情況進行配置。

nginx和php-fpm在同一臺伺服器上,這時可以直接用unix socket程式間通訊,不走tcp埠通訊,可以節約建立連線的時間,從而提高效能。sock檔案隨便建立到哪裡都可以,只要fpm有許可權在那個目錄裡寫檔案,nginx有許可權去讀就可以。tcp連線會更穩定,因為有tcp協議保證資料的正確性,但是sock有更少的資料拷貝和上下文切換,更少的資源佔用。不過只能在nginx和fpm在同一臺機器上才能用socket。

如何選擇socket啟動還是埠啟動。

由於tcp方式相對unix的方式,併發量更高,因此針對併發量高的專案,建議採用tcp方式,現在Nginx配置示例檔案預設的也是tcp方式。

使用unix方式,可以優化的點,就是將socket檔案放在/dev/shm目錄下面,至於為什麼放在這個目錄可以參考.www.linuxidc.com/Linux/2014-05/101... 。大致的意思,就是該目錄下面的檔案是不是儲存再硬碟中的,而是儲存再記憶體中的。至於硬碟讀取和記憶體讀取,誰快誰慢,肯定是記憶體最快了。

socket方式啟動如何檢視socke檔案。

socket檔案是根據上面提到的pid配置項而定的。我們可以直接使用cat命令,檢視程式號。

子程式預設啟動數量,通過上面的pm = dynamic 配置,我們知道這種方式是動態配置子程式大小的,同時我們也可以設定預設的子程式數。

pm = dynamic
pm.max_children = 20
### 預設15個子程式,演示的效果就是上面的shell命令的結果圖。
pm.start_servers = 15 

當我們嘗試設定為3時,顯示如下錯誤資訊。

說明,這裡的start_servers配置項和min_spare_servers配置是有一定的關係的。我們設定為最小10,結果就能正常啟動php-fpm了。

本作品採用《CC 協議》,轉載必須註明作者和本文連結

相關文章