php-fpm是什么
php-fpm是PHP的一個進(jìn)程管理器。php下面的眾多work進(jìn)程皆有php-fpm進(jìn)程管理器管理。
php-fpm的工作原理
php-fpm全名是PHP FastCGI進(jìn)程管理器。php-fpm啟動后會先讀php.ini,然后再讀相應(yīng)的conf配置文件,conf配置可以覆蓋php.ini的配置。
啟動php-fpm之后,會創(chuàng)建一個master進(jìn)程,監(jiān)聽9000端口(可配置),master進(jìn)程又會根據(jù)fpm.conf/www.conf去創(chuàng)建若干子進(jìn)程,子進(jìn)程用于處理實(shí)際的業(yè)務(wù)。
當(dāng)有客戶端(比如nginx)來連接9000端口時,空閑子進(jìn)程會自己去accept,如果子進(jìn)程全部處于忙碌狀態(tài),新進(jìn)的待accept的連接會被master放進(jìn)隊列里,等待fpm子進(jìn)程空閑;這個存放待accept的半連接的隊列有多長,由listen.backlog 配置。
如何查看php-fpm進(jìn)程與子進(jìn)程
查看php-fpm相關(guān)的所有進(jìn)程。如下圖
![](http://img.jbzj.com/file_images/article/202006/20206185055119.png?20205185243)
這里pool www
皆是php-fpm的子進(jìn)程,也就是我們常說的work進(jìn)程。
查看php-fpm下面的子進(jìn)程
通過上面的命令,其實(shí)我們能夠看出php-fpm相關(guān)的進(jìn)程了,如果我們需要更加直觀的查看php-fpm的master進(jìn)程和work進(jìn)程,可以通過下面的方式進(jìn)程查看。
這里的5370則是php-fpm的master進(jìn)程號。通過上面的命令已經(jīng)很能直觀的得出。
![](http://img.jbzj.com/file_images/article/202006/20206185650083.png?20205185716)
通過上面的命令,可以看出php-fpm作為master進(jìn)程,下面有15個子進(jìn)程。這里的子進(jìn)程數(shù)都是可以進(jìn)程自定義配置。通過如下幾個參數(shù)進(jìn)程配置:
pm = dynamic # 動態(tài)創(chuàng)建子進(jìn)程
pm.max_children = 20 # 最大子進(jìn)程數(shù)
pm.start_servers = 15 # 初始化php-fpm進(jìn)程時,默認(rèn)的子進(jìn)程數(shù)
php-fpm參數(shù)配置說明
php-fpm全局配置參數(shù)
#php-fpm的運(yùn)行權(quán)限。
#以什么用戶什么組的權(quán)限來運(yùn)行池fpm。
user = www
group = www
#php-fpm的運(yùn)行方式,可以使端口,也可以使socke文件。
#如果是端口則是走tcp,如果是socket則直接讀socket文件,這樣性能相對更好。
listen = 127.0.0.1:9000
#擁有socket權(quán)限的用戶,需要和上面的user、group配置相結(jié)合。
#如果采用的端口的方式,則不配置。
listen.owner = www
listen.group = www
listen.mode = 0660
#這是php-fpm端口連接的地址。多個用","隔開。默認(rèn)任意地址都可以連接。
#例如Nginx和php-fpm不在同一臺服務(wù)器上,這里的值就是Nginx服務(wù)的ip地址。
#當(dāng)Nginx和php-fpm配置在同一臺服務(wù)器上,則直接寫127.0.0.1即可。
listen.allowed_clients = 127.0.0.1
#pid進(jìn)程文件存放的位置,當(dāng)我們啟用一個php服務(wù),
#則會自動創(chuàng)建一個該pid文件,其實(shí)我們可以直接把該文件理解理解php-fpm的進(jìn)程號文件,
#兩則是等價的。默認(rèn)為none。
pid = /opt/remi/php72/root/var/run/php-fpm/php-fpm.pid
#錯誤日志位置,默認(rèn):安裝路徑 #INSTALL_PREFIX#/log/php-fpm.log。
#如果設(shè)置為syslog,log就會發(fā)送給syslogd服務(wù)而不會寫進(jìn)文件里。
error_log = /opt/remi/php72/root/var/log/php-fpm/error.log
#PHP限制的文件擴(kuò)展名
security.limit_extensions = .php .php3 .php4 .php5 .php7
#系統(tǒng)日志標(biāo)示,如果跑了多個fpm進(jìn)程,需要用這個來區(qū)分日志是誰的。
syslog.ident = php-fpm
#日記登記,可選:alert, error, warning, notice, debug。
log_level = notice
#緊急重啟閾值,需要與下面emergency_restart_interval參數(shù)一起配置。
emergency_restart_threshold = 60
# 緊急重啟閾值的時間范圍。在此參數(shù)設(shè)置的時間內(nèi),
# 出現(xiàn)SIGSEGV或SIGBUS的子進(jìn)程數(shù)超過emergency_restart_threshold參數(shù)設(shè)置的值。
# 那么fpm就會優(yōu)雅的重啟,值是0表示off這個功能,可用的單位有:s秒,m分,h時,d天。
emergency_restart_interval = 60s
#設(shè)置子進(jìn)程接受主進(jìn)程復(fù)用信號的超時時間。
process_control_timeout = 0
#當(dāng)動態(tài)管理子進(jìn)程時,fpm最多能fork多少個進(jìn)程,0表示無限制,
# 這是所有進(jìn)程池能啟動子進(jìn)程的總和,謹(jǐn)慎使用。
process.max = 128
#設(shè)置子進(jìn)程的優(yōu)先級,在master進(jìn)程以root用戶啟動時有效;
#如果沒有設(shè)置,子進(jìn)程會繼承master進(jìn)程的優(yōu)先級,值范圍-19(最高)到20(最低),默認(rèn)不設(shè)置。
process.priority = -19
#設(shè)置成no用于調(diào)試bug,默認(rèn)為yes。
daemonize = yes
#master進(jìn)程最多能打開的文件數(shù)量。默認(rèn)采用系統(tǒng)設(shè)置的值。
rlimit_files = 1024
#master進(jìn)程核心rlimit限制值;可選unlimited或>=0的整數(shù),默認(rèn)為系統(tǒng)的值。
rlimit_core = 0
#事件處理機(jī)制,默認(rèn)自動檢測,可選值:select,poll,
#epoll(linux>=2.5.44),kqueue,/dev/poll,port
events.mechanism = epoll
#fpm想系統(tǒng)發(fā)送狀態(tài)的頻率。單位有s,m,h。
#前提是fpm被設(shè)置會系統(tǒng)服務(wù)。
systemd_interval = 10s
php-fpm的進(jìn)程進(jìn)程池配置
#php-fpm的隊列長度。
listen.backlog = 65535
#php進(jìn)程池權(quán)限,同樣要master進(jìn)程是root用戶才有效,
#和上面的全局設(shè)置一樣,不設(shè)置的話會繼承master進(jìn)程的優(yōu)先級。
process.priority = -19
#子進(jìn)程管理方式
#static(靜態(tài)配置,在啟動php-fpm時根據(jù)該值創(chuàng)建固定的子進(jìn)程數(shù)量);
#dynamic(動態(tài)配置,在啟動php-fpm時根據(jù)pm.start_servers的值初始化對應(yīng)的子進(jìn)程數(shù),至少一個子進(jìn)程);
#ondemand(按需配置,在啟動php-fpm時不創(chuàng)建子進(jìn)程,而是根據(jù)請求動態(tài)fork子進(jìn)程);
pm = dynamic
#最大子進(jìn)程數(shù)量
pm.max_children = 5
#初始化子進(jìn)程數(shù)量,與上面的pm = dynamic配置使用。
pm.start_servers = 2
#服務(wù)器閑置時最少保持2個子進(jìn)程,不夠這個數(shù)就會創(chuàng)建,只適用動態(tài)dynamic管理方式
pm.min_spare_servers = 2
#服務(wù)器閑置時最多要有幾個,多了會kill,只適用動態(tài)dynamic管理方式
pm.max_spare_servers = 3
#子進(jìn)程閑置時間,也就是說子進(jìn)程沒有可處理的任務(wù)時,在該之間使就會被killed。
pm.process_idle_timeout = 10s
#每個子進(jìn)程最大的處理請求數(shù)量。在一定程度上可以防止內(nèi)存泄漏。
pm.max_requests = 500
#php-fpm狀態(tài)監(jiān)控的uri
pm.status_path string
#php-fpm監(jiān)控頁面的 ping 網(wǎng)址。
#如果沒有設(shè)置,則無法訪問 ping 頁面。
#該頁面用于外部檢測php-fpm是否存活并且可以響應(yīng)請求。請注意必須以斜線開頭(/)。
ping.path string
#用于定義ping請求的返回響應(yīng)。返回為 HTTP 200 的 text/plain 格式文本。默認(rèn)值:pong。
ping.response string
#設(shè)置worker的nice(2)優(yōu)先級(如果設(shè)置了的話)。
#該值從 -19(最高優(yōu)先級) 到 20(更低優(yōu)先級)。
#默認(rèn)值:不設(shè)置
process.priority int
#檢測路徑時使用的前綴
prefix string
#訪問文件日志,沒啥用處,比如yii2每次都記錄訪問index.php,只是記錄真實(shí)的PHP文件。
access.log = var/log/$pool.access.log
#php的慢日志
slowlog = var/log/$pool.log.slow
#慢日志時間閾值
request_slowlog_timeout = 2s
#單個請求的超時時間,當(dāng)php.ini設(shè)置的最大執(zhí)行時間未生效,則交由它來處理。
request_terminate_timeout = 3s
#最大打開句柄數(shù),默認(rèn)為系統(tǒng)值。
rlimit_files = 1024
#最多的核心使用數(shù),默認(rèn)為系統(tǒng)分配。
rlimit_core = 0
部分配置演示
php-fpm的backlog大小設(shè)置
php-fpm的backlog大小設(shè)置與php-fpm的處理能力有關(guān),而不是越大越好。
當(dāng)該值設(shè)置過大,導(dǎo)致php-fpm處理不過來,nginx那邊等待超時,斷開連接,報504 gateway timeout錯。同時php-fpm處理完準(zhǔn)備write 數(shù)據(jù)給nginx時,發(fā)現(xiàn)TCP連接斷開了,報“Broken pipe”。
當(dāng)該值設(shè)置過小,nginx之類的client請求,根本進(jìn)入不了php-fpm的accept queue,報“502 Bad Gateway”錯。所以,這還得去根據(jù)php-fpm的QPS來決定backlog的大小。計算方式最好為QPS=backlog。
php-fpm啟動模式
php-fpm以socket啟動或者端口啟動,這兩種的方式根據(jù)實(shí)際情況進(jìn)行配置。
nginx和php-fpm在同一臺服務(wù)器上,這時可以直接用unix socket進(jìn)程間通信,不走tcp端口通信,可以節(jié)約創(chuàng)建連接的時間,從而提高性能。sock文件隨便創(chuàng)建到哪里都可以,只要fpm有權(quán)限在那個目錄里寫文件,nginx有權(quán)限去讀就可以。tcp連接會更穩(wěn)定,因?yàn)橛衪cp協(xié)議保證數(shù)據(jù)的正確性,但是sock有更少的數(shù)據(jù)拷貝和上下文切換,更少的資源占用。不過只能在nginx和fpm在同一臺機(jī)器上才能用socket。
如何選擇socket啟動還是端口啟動。
由于tcp方式相對unix的方式,并發(fā)量更高,因此針對并發(fā)量高的項(xiàng)目,建議采用tcp方式,現(xiàn)在Nginx配置示例文件默認(rèn)的也是tcp方式。
使用unix方式,可以優(yōu)化的點(diǎn),就是將socket文件放在/dev/shm目錄下面,大致的意思,就是該目錄下面的文件是不是存儲再硬盤中的,而是存儲再內(nèi)存中的。至于硬盤讀取和內(nèi)存讀取,誰快誰慢,肯定是內(nèi)存最快了。
socket方式啟動如何查看socke文件。
socket文件是根據(jù)上面提到的pid配置項(xiàng)而定的。我們可以直接使用cat命令,查看進(jìn)程號。
![](http://img.jbzj.com/file_images/article/202006/20206190651775.png?2020519724)
子進(jìn)程默認(rèn)啟動數(shù)量,通過上面的pm = dynamic 配置,我們知道這種方式是動態(tài)配置子進(jìn)程大小的,同時我們也可以設(shè)置默認(rèn)的子進(jìn)程數(shù)。
pm = dynamic
pm.max_children = 20
### 默認(rèn)15個子進(jìn)程,演示的效果就是上面的shell命令的結(jié)果圖。
pm.start_servers = 15
當(dāng)我們嘗試設(shè)置為3時,顯示如下錯誤信息。
![](http://img.jbzj.com/file_images/article/202006/20206190905517.png?2020519937)
說明,這里的start_servers配置項(xiàng)和min_spare_servers配置是有一定的關(guān)系的。我們設(shè)置為最小10,結(jié)果就能正常啟動php-fpm了。
以上就是一文看懂PHP進(jìn)程管理器php-fpm的詳細(xì)內(nèi)容,更多關(guān)于PHP進(jìn)程管理器php-fpm的資料請關(guān)注腳本之家其它相關(guān)文章!
您可能感興趣的文章:- 淺談并發(fā)處理PHP進(jìn)程間通信之外部介質(zhì)
- PHP如何限制定時任務(wù)的進(jìn)程數(shù)量
- PHP基于進(jìn)程控制函數(shù)實(shí)現(xiàn)多線程
- php 的多進(jìn)程操作實(shí)踐案例分析
- php 多進(jìn)程編程父進(jìn)程的阻塞與非阻塞實(shí)例分析
- php實(shí)現(xiàn)的簡單多進(jìn)程服務(wù)器類完整示例
- PHP 進(jìn)程池與輪詢調(diào)度算法實(shí)現(xiàn)多任務(wù)的示例代碼
- php進(jìn)程(線程)通信基礎(chǔ)之System V共享內(nèi)存簡單實(shí)例分析
- 淺談并發(fā)處理PHP進(jìn)程間通信之System V IPC