您好,登錄后才能下訂單哦!
先來了解一些名詞概念:
CGI是Common Gateway Interface(通用網管協議),用于讓交互程序和Web服務器通信的協議。它負責處理URL的請求,啟動一個進程,將客戶端發送的數據作為輸入,由Web服務器收集程序的輸出并加上合適的頭部,再發送回客戶端。
FastCGI是基于CGI的增強版本的協議,不同于創建新的進程來服務請求,使用持續的進程和創建的子進程來處理一連串的進程,這些進程由FastCGI服務器管理,開銷更小,效率更高。
PHP-FPM是PHP實現的FastCGI Process Manager(FastCGI進程管理器), 用于替換PHP FastCGI的大部分附加功能,適用于高負載網站。支持的功能如:
平滑停止/啟動的高級進程管理功能
慢日志記錄腳本
動態/靜態子進程產生
基于php.ini的配置文件
PHP-FPM在5.4之后已經整合進入PHP源代碼中,提供更好的PHP進程管理方式,可以有效控制內存和進程,平滑重載PHP配置。如果需要使用,在./configure的時候帶上-enable-fpm參數即可,使用PHP-FPM來控制FastCGI進程:
// 支持start/stop/quit/restart/reload/logrotate參數
// quit/reload是平滑終止和平滑重新加載,即等現有的服務完成
./php-fpm --start
PHP-FPM 配置
PHP-FPM配置文件為php-fpm.conf,在這個配置文件中我們需要了解一些參數。下面所有的子進程均指php-fpm進程,可以在終端通過ps aux | grep php查看到。
顯示php-fpm: pool www的代表work子進程(實際處理請求)
顯示php-fpm: process master的代表master主進程(負責管理work子進程)
全局配置
先看PHP-FPM最重要的全局配置部分:
emergency_restart_threshold
如果在emergency_restart_interval設定的時間內收到該參數設定次數的SIGSEGV或SIGBUS退出的信號,則FPM會重新啟動。默認值為0,表示關閉該功能。
emergency_restart_interval
設定平滑重啟的間隔時間,有助于解決加速器中共享內存的使用問題。可用單位s(默認)/m/h/d,默認值為0, 表示關閉。
process.max
FPM能夠創建的最大子進程數量,它在使用多個pm = dynamic配置的php-fpm pool進程池的時候,控制全局的子進程數量。默認值為0,代表著無限制。
進程池配置
PHP-FPM的配置其余部分是一個名為Pool Definitions的區域,這個區域的配置設置每個PHP-FPM進程池,進程池中是一系列相關的子進程。這部分開頭都是[進程池名稱],如[www]。
此時可以解釋看到ps aux | grep php中顯示的是php-fpm: pool www。
pm
pm指的是process manager,指定進程管理器如何控制子進程的數量,它為必填項,支持3個值:
static: 使用固定的子進程數量,由pm.max_children指定
dynamic:基于下面的參數動態的調整子進程的數量,至少有一個子進程
pm.max_chidren: 可以同時存活的子進程的最大數量
pm.start_servers: 啟動時創建的子進程數量,默認值為min_spare_servers + max_spare_servers - min_spare_servers) / 2
pm.min_spare_servers: 空閑狀態的子進程的最小數量,如果不足,新的子進程會被自動創建
pm.max_spare_servers: 空閑狀態的子進程的最大數量,如果超過,一些子進程會被殺死
ondemand: 啟動時不會創建子進程,當新的請求到達時才創建。會使用下面兩個參數:
pm.max_children
pm.process_idle_timeout 子進程的空閑超時時間,如果超時時間到沒有新的請求可以服務,則會被殺死
pm.max_requests
每一個子進程的最大請求服務數量,如果超過了這個值,該子進程會被自動重啟。在解決第三方庫的內存泄漏問題時,這個參數會很有用。默認值為0,指子進程可以持續不斷的服務請求。
PHP-FPM配置優化
PHP-FPM管理的方式是一個master主進程,多個pool進程池,多個worker子進程。其中每個進程池監聽一個socket套接字。具體的圖示:
圖片描述
其中的worker子進程實際處理連接請求,master主進程負責管理子進程:
master
進程,設置1s定時器,通過socket
文件監聽pm=dynamic
時,如果idle worker
數量<pm.min_spare_servers
,創建新的子進程pm=dynamic
時,如果idle worker
數量>pm.max_spare_servers
,殺死多余的空閑子進程pm=ondemand
時,如果idle worker
空閑時間>pm.process_idle_timeout
,殺死該空閑進程worker
數量>pm.max_children
,打印warning
日志,退出;如果無異常,使用idle worker
服務,或者新建worker
服務我們為了避免PHP-FPM主進程由于某些糟糕的PHP代碼掛掉,需要設置重啟的全局配置:
; 如果在1min內有10個子進程被中斷失效,重啟主進程
emergency_restart_threshold = 10
emergency_restart_interval = 1m
進程數調優
每一個子進程同時只能服務一次連接,所以控制同時存在多少個進程數就很重要,如果過少會導致很多不必要的重建和銷毀的開銷,如果過多又會占用過多的內存,影響其他服務使用。
我們應該測試自己的PHP進程使用多少內存,一般來說剛啟動時是8M左右,運行一段時間由于內存泄漏和緩存會上漲到30M左右,所以你需要根據自己的預期內存大小設定進程的數量。同時根據進程池的數量來看一個進程管理器的子進程數量限制。
測試平均PHP子進程占用的內存:
$ps auxf | grep php | grep -v grep
work 26829 0.0 0.0 715976 4712 ? Ss Jul11 0:00 php-fpm: master process (./etc/php-fpm.conf)
work 21889 0.0 0.0 729076 29668 ? S 03:12 0:20 _ php-fpm: pool www
work 21273 0.0 0.0 728928 31380 ? S 03:25 0:21 _ php-fpm: pool www
work 15114 0.0 0.0 728052 29084 ? S 03:40 0:19 _ php-fpm: pool www
work 17072 0.0 0.0 728800 34240 ? S 03:54 0:22 _ php-fpm: pool www
work 22763 0.0 0.0 727904 20352 ? S 11:29 0:04 _ php-fpm: pool www
work 38545 0.0 0.0 727796 19484 ? S 12:34 0:01 _ php-fpm: pool www
// 共占用的內存數量
$ps auxf | grep php | grep -v grep | grep -v master | awk '{sum+=$6} END {print sum}'
162712
// 所有的子進程數量
$ ps auxf | grep php | grep -v grep | grep -v master | wc -l
6
可以看到第6列,每一個子進程的內存占用大概在19-34M之間(單位為KB)。平均的內存占用為162712KB/6 = 27.1M。
查看服務器總的內存大小
$ free -g
total used free shared buffers cached
Mem: 157 141 15 0 4 123
-/+ buffers/cache: 13 143
Swap: 0 0 0
可以看出我的服務器總得內存大小是157G(-g采用了G的單位)。
進程數限制
此時如果我們分配全部的內存給PHP-FPM使用,那么進程數可以限制在157000/27 = 5814,但是由于我的服務器同時服務了很多內容,所以我們可以向下調整到512個進程數:
process.max = 512
pm = dynamic
pm.max_children = 512
pm.start_servers = 16
pm.min_spare_servers = 8
pm.max_spare_serveres = 30
防止內存泄漏
由于糟糕的插件和庫,內存泄漏時有發生,所以我們需要對每一個子進程服務的請求數量做限制,防止無限制的內存泄漏:
pm.max_requests = 1000
重啟
如果上面的配置都按照你的實際需求和環境配置好了,不要忘記重啟PHP-FPM服務。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。