nginx之php-fpm優化

語言: CN / TW / HK

先來看一段日誌(此日誌來自於 var/log/php-fpm.log ):

[13-Aug-2017 03:30:03] NOTICE: fpm is running, pid 28263
[13-Aug-2017 03:30:03] NOTICE: ready to handle connections
[13-Aug-2017 10:11:04] WARNING: [pool www] seems busy (you may need to increase pm.start_servers, or pm.min/max_spare_servers), spawning 8 children, there are 0 idle, and 7 total children
[13-Aug-2017 10:11:05] WARNING: [pool www] seems busy (you may need to increase pm.start_servers, or pm.min/max_spare_servers), spawning 16 children, there are 0 idle, and 8 total children
[13-Aug-2017 10:11:06] WARNING: [pool www] seems busy (you may need to increase pm.start_servers, or pm.min/max_spare_servers), spawning 32 children, there are 0 idle, and 9 total children
[13-Aug-2017 10:11:07] WARNING: [pool www] seems busy (you may need to increase pm.start_servers, or pm.min/max_spare_servers), spawning 32 children, there are 0 idle, and 10 total children
[13-Aug-2017 10:11:08] WARNING: [pool www] seems busy (you may need to increase pm.start_servers, or pm.min/max_spare_servers), spawning 32 children, there are 0 idle, and 11 total children
[13-Aug-2017 10:11:09] WARNING: [pool www] seems busy (you may need to increase pm.start_servers, or pm.min/max_spare_servers), spawning 32 children, there are 0 idle, and 12 total children
[13-Aug-2017 10:11:10] WARNING: [pool www] seems busy (you may need to increase pm.start_servers, or pm.min/max_spare_servers), spawning 32 children, there are 0 idle, and 13 total children
[13-Aug-2017 10:11:11] WARNING: [pool www] seems busy (you may need to increase pm.start_servers, or pm.min/max_spare_servers), spawning 32 children, there are 0 idle, and 14 total children

一、首先分析驗證問題:

1.報警級別“WARNING”,也就是說業務受到的影響比較小,實際驗證也是這樣;

2.php-fpm的程序數比較少,建議增加 pm.start_servers, or pm.min/max_spare_servers

3.增加到多少呢?spawning 8,也就是說至少得8個程序。

1.2 查詢一個php佔用的記憶體方法:

pmap $(pgrep php-fpm | head -1)

二、php-fpm的配置公式:

pm.start_servers = min_spare_servers + (max_spare_servers - min_spare_servers) / 2

三、php-fpm優化方法:

php-fpm存在兩種方式,一種是直接開啟指定數量的php-fpm程序,不再增加或者減少—靜態;

另一種則是開始時開啟一定數量的php-fpm程序,當請求量變大時,動態的增加php-fpm程序數到上限,當空閒時自動釋放空閒的程序數到一個下限—動態。

這兩種不同的執行方式,可以根據伺服器的實際需求來進行調整。

要用到的一些引數,分別是pm、pm.max_children、pm.start_servers、pm.min_spare_servers和pm.max_spare_servers。

pm表示使用那種方式,有兩個值可以選擇,就是static(靜態)或者dynamic(動態)。

3.1 下面4個引數的意思分別為:

pm.max_children:靜態方式下開啟的php-fpm程序數量;在動態方式下他限定php-fpm的最大程序數(這裡要注意pm.max_spare_servers的值只能小於等於pm.max_children)

pm.start_servers:動態方式下的起始php-fpm程序數量。

pm.min_spare_servers:動態方式空閒狀態下的最小php-fpm程序數量。

pm.max_spare_servers:動態方式空閒狀態下的最大php-fpm程序數量。

如果dm設定為static,那麼其實只有pm.max_children這個引數生效。系統會開啟設定的數量個php-fpm程序,始終保持一個固定數量的子程序,這個數由pm.max_children定義,這種方式很不靈活,也通常不是預設的。

如果dm設定為dynamic,4個引數都生效。系統會在php-fpm執行開始時啟動pm.start_servers個php-fpm程序,他是這樣的,啟動時,會產生固定數量的子程序(由pm.start_servers控制)可以理解成最小子程序數,而最大子程序數則由pm.max_children去控制,OK,這樣的話,子程序數會在最大和最小數範圍中變化,還沒有完,閒置的子程序數還可以由另2個配置控制,分別是pm.min_spare_servers和pm.max_spare_servers,也就是閒置的子程序也可以有最小和最大的數目,而如果閒置的子程序超出了pm.max_spare_servers,則會被殺掉。

可以看到,pm = dynamic模式非常靈活,也通常是預設的選項。但是,dynamic模式為了最大化地優化伺服器響應,會造成更多記憶體使用,因為這種模式只會殺掉超出最大閒置程序數(pm.max_spare_servers)的閒置程序,比如最大閒置程序數是30,最大程序數是50,然後網站經歷了一次訪問高峰,此時50個程序全部忙碌,0個閒置程序數,接著過了高峰期,可能沒有一個請求,於是會有50個閒置程序,但是此時php-fpm只會殺掉20個子程序,始終剩下30個程序繼續作為閒置程序來等待請求,這可能就是為什麼過了高峰期後即便請求數大量減少伺服器記憶體使用卻也沒有大量減少,也可能是為什麼有些時候重啟下伺服器情況就會好很多,因為重啟後,php-fpm的子程序數會變成最小閒置程序數,而不是之前的最大閒置程序數。

第三種就是pm = ondemand模式,這種模式和pm = dynamic相反,把記憶體放在第一位,他的工作模式很簡單,每個閒置程序,在持續閒置了pm.process_idle_timeout秒後就會被殺掉,有了這個模式,到了伺服器低峰期記憶體自然會降下來,如果伺服器長時間沒有請求,就只會有一個php-fpm主程序,當然弊端是,遇到高峰期或者如果pm.process_idle_timeout的值太短的話,無法避免伺服器頻繁建立程序的問題,因此pm = dynamic和pm = ondemand誰更適合視實際情況而定。

那麼,對於伺服器,選擇哪種執行方式比較好呢?事實上,跟Apache一樣,執行的PHP程式在執行完成後,或多或少會有記憶體洩露的問題。這也是為什麼開始時一個php-fpm程序只佔用3M左右記憶體,執行一段時間後就會上升到25-40M的原因了。

所以,動態方式因為會結束掉多餘的程序,可以回收釋放一些記憶體,所以推薦在記憶體較少的伺服器或者VPS上使用。具體最大數量根據 記憶體/25 得到。

比如說512M的VPS,建議pm.max_spare_servers設定為20(512*0.8/20)。至於pm.min_spare_servers,則建議根據伺服器的負載情況來設定,比較合適的值在5~10之間。

然後對於比較大記憶體的伺服器來說,設定為靜態的話會提高效率。

因為頻繁開關php-fpm程序也會有時滯,所以記憶體夠大的情況下開靜態效果會更好。數量也可以根據 記憶體/30M 得到。

比如說2GB記憶體的伺服器,可以設定為50;4GB記憶體可以設定為100等。

比如,我的1024M的騰訊雲,設定的引數如下:

pm = dynamic
pm.max_children = 35
pm.start_servers = 8
pm.min_spare_servers = 8
pm.max_spare_servers = 35

可以最大的節省記憶體並提高執行效率。