php-fpm占用cpu和內存過高100% 解決辦法


 

參考網站:

https://www.fujieace.com/php/php-fpm.html

 

https://www.fujieace.com/php/pm-max_children-2.html

 

pm.max_children 設置多大合適?

php-fpm.conf有兩個至關重要的參數:

一個是”max_children”,另一個是”request_terminate_timeout”。

 

pm.max_children 表示 php-fpm 能啟動的子進程的最大數量。

request_terminate_timeout 表示將執行時間太長的進程直接終止。

我的兩個設置的值一個是”40″,一個是”900″,但是這個值不是通用的,而是需要自己計算的。

 

一、pm.max_children 多大合適?

這個值原則上是越大越好,php-cgi的進程多了就會處理的很快,排隊的請求就會很少。

設置”max_children” 也需要根據服務器的性能進行設定。

 

計算方式如下:

一般來說一台服務器正常情況下每一個php-cgi所耗費的內存在20M~30M左右,因此我的”max_children”我設置成40個,20M*40=800M也就是說在峰值的時候所有PHP-CGI所耗內存在800M以內,低於我的有效內存2Gb。

而如果我 的”max_children”設置的較小,比如5-10個,那么php-cgi就會“很累“,處理速度也很慢,等待的時間也較長,占用的CPU也很高。

如果長時間沒有得到處理的請求就會出現 504 Gateway Time-out 這個錯誤,而正在處理的很累的那幾個php-cgi如果遇到了問題就會出現 502 Bad gateway 這個錯誤。

 

max_children較好的設置方式根據req/s(吞吐率,單位時間里服務器處理的最大請求數,單位req/s)來設置,若程序是 100 req/s 的處理能力,那么就設置 100比較好,這是動態來調整的。

 

二、request_terminate_timeout 多大合適?

計算方式如下:

如果你的服務器性能足夠好,且寬帶資源足夠充足,PHP腳本沒有循環或BUG的話你可以直接將”request_terminate_timeout”設 置成0s。0s的含義是讓PHP-CGI一直執行下去而沒有時間限制。

而如果你做不到這一點,也就是說你的PHP-CGI可能出現某個BUG,或者你的寬帶不夠充足或者其他的原因導致你的PHP-CGI能夠假死那么就建議你給”request_terminate_timeout”賦一個值,這個值可以根 據你服務器的性能進行設定。

一般來說性能越好你可以設置越高,20分鍾-30分鍾都可以。由於我的服務器PHP腳本需要長時間運行,有的可能會超過10分鍾因此我設置了900秒,這樣不會導致PHP-CGI死掉而出現502 Bad gateway這個錯誤。

 

 

#################################################################################

 

pm = dynamic; 表示使用哪種進程數量管理方式

 

dynamic表示php-fpm進程數是動態的,最開始是pm.start_servers指定的數量,如果請求較多,則會自動增加,保證空閑的進程數不小於pm.min_spare_servers,如果進程數較多,也會進行相應清理,保證多余的進程數不多於pm.max_spare_servers;

 

static表示php-fpm進程數是靜態的,進程數自始至終都是pm.max_children指定的數量,不再增加或減少。

pm.max_children = 300; 靜態方式下開啟的php-fpm進程數量

pm.start_servers = 20; 動態方式下的起始php-fpm進程數量

pm.min_spare_servers = 5; 動態方式下的最小php-fpm進程數量

pm.max_spare_servers = 35; 動態方式下的最大php-fpm進程數量

 

數值設置,參考自己的實際硬件配置,可以參考 總內存/30M 來計算。

 

如果dm設置為static,那么其實只有pm.max_children這個參數生效。系統會開啟設置數量的php-fpm進程。

如果dm設置為dynamic,那么pm.max_children參數失效,后面3個參數生效。系統會在php-fpm運行開始的時候啟動pm.start_servers個php-fpm進程,然后根據系統的需求動態在pm.min_spare_servers和pm.max_spare_servers之間調整php-fpm進程數。

 

如何判斷我選擇“pm = dynamic”還是“pm = static”呢?哪一種更好呢?

事實上,跟Apache一樣,運行的PHP程序在執行完成后,或多或少會有內存泄露的問題。

這也是為什么開始的時候一個php-fpm進程只占用3M左右內存,運行一段時間后就會上升到20-30M的原因了。

 

對於內存大的服務器(比如8G以上)來說,用靜態的max_children實際上更為妥當,因為這樣不需要進行額外的進程數目控制,會提高效率。因為頻繁開關php-fpm進程也會有時滯,所以內存夠大的情況下開靜態效果會更好。數量也可以根據 總內存/30M 得到,比如8GB內存可以設置為100,那么php-fpm耗費的內存就能控制在 2G-3G的樣子。

如果內存稍微小點,比如1~2G,那么指定靜態的進程數量更加有利於服務器的穩定。這樣可以保證php-fpm只獲取夠用的內存,將不多的內存分配給其他應用去使用,會使系統的運行更加暢通。

 

對於小內存的服務器來說,比如256M內存的VPS,即使按照一個20M的內存量來算,10個php-cgi進程就將耗掉200M內存,那系統的崩潰就應該很正常了。

因此應該盡量地控制php-fpm進程的數量,大體明確其他應用占用的內存后,給它指定一個靜態的小數量,會讓系統更加平穩一些。

或者使用動態方式,因為動態方式會結束掉多余的進程,可以回收釋放一些內存,所以推薦在內存較少的服務器或VPS上使用,具體最大數量根據 總內存/20M 得到。

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

 

總結:內存小的建議用動態(pm = dynamic),內存大的建議用靜態(pm = static)。

 

配置php慢日志,用於監控

request_slowlog_timeout = 10s

slowlog = log/$pool.log.slow

 

配置php-fpm進程可打開的最大文件句柄數,

rlimit_files = 1024

默認1024,此值可以不需要配置


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM