避免 php fpm 耗盡記憶體導致宕機

2021-12-30 07:05:19 字數 2221 閱讀 1863

昨天晚上php-fpm出錯

首先檢查下個服務是否處於啟動狀態,由於我的 mysql 還是在另一台伺服器上放著,用 ss -tln 命令檢視後確認服務都處於正常開啟狀態,然後再看伺服器的負載情況:

確認是不是由於負載過大而引起的,然而並不是,像我這種小站一般不會出現負載過大的情況。

繼續檢視是否由於記憶體耗盡或頻寬限制等原因,用 free -m 檢視後發現伺服器的記憶體已經耗盡了:

繼續查詢記憶體耗盡的原因:

可以看出是由於 php-fpm 有好多閒置的程序一直不釋放, 導致記憶體占用過大,然後開始尋求解決辦法,經過一番查詢發現,fastcgi 程序一旦載入就不會釋放,當其工作完成後,就休眠於 fastcgi 系統池中,等待下一次被喚醒。我一直用的是 php-fpm 預設配置,這個配置對於我來說可能有點不合適,需要修改配置檔案。下面是一些優化的技巧:

1、php-fpm優化引數介紹

分別有:pm、pm.max_children、pm.start_servers、pm.min_spare_servers、pm.max_spare_servers。

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

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

pm.max_children:靜態方式下開啟的php-fpm程序數量

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

pm.min_spare_servers:動態方式下的最小php-fpm程序數

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

區別:如果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程序數

選擇哪種執行方式比較好呢?事實上,跟apache一樣,執行的php程式在執行完成後,或多或少會有記憶體洩露的問題。根據網上的資料,對於記憶體大的伺服器(8g)以上,指定靜態的max_children實際上更為妥當,因為這樣不需要進行額外的程序數目控制,會提高效率。對於小記憶體的伺服器,動態方式會結束掉多餘的程序,可以**釋放一些記憶體,所以推薦在記憶體較少的伺服器或vps上使用。

php-fpm 預設的配置是這樣的:

[root@feiyu etc]# cat php-fpm.conf | grep -v ';' | grep -v '^$' [global] pid=run/php-fpm.pid [www] user=nobody group=nobody listen=127.0.0.1:9000 pm=dynamic pm.max_children=20 pm.start_servers=10 pm.min_spare_servers=5 pm.max_spare_servers=20 pm.max_requests=500

我調整後的配置:

[global] pid=run/php-fpm.pid [www] user=nobody group=nobody listen=127.0.0.1:9000 pm=dynamic pm.max_children=20 pm.start_servers=5 pm.min_spare_servers=2 pm.max_spare_servers=10 pm.max_requests=300

然後重啟 php-fpm ,重啟完記憶體使用是這樣的,通過上面的對比可以看出 php-fpm 消耗了不少記憶體:

服務執行穩定以後系統的記憶體使用是這樣的:

記憶體耗盡怎麼辦?

如果在申請動態記憶體時找不到足夠大的記憶體塊,malloc和new將返回null指標,宣告記憶體申請失敗。通常有三種方式處理 記憶體耗盡 問題。1 判斷指標是否為null,如果是則馬上用return語句終止本函式。例如 void func void 2 判斷指標是否為null,如果是則馬上用exit...

7 9 記憶體耗盡怎麼辦?

如果在申請動態記憶體時找不到足夠大的記憶體塊,malloc 和new 將返回null 指標,宣告記憶體申請失敗。通常有三種方式處理 記憶體耗盡 問題。1 判斷指標是否為null,如果是則馬上用return 語句終止本函式。例如 void func void 2 判斷指標是否為null,如果是則馬上用...

C 記憶體耗盡怎麼辦?

如果在申請動態記憶體時找不到足夠大的記憶體塊,malloc和new將返回null指標,宣告記憶體申請失敗。通常有三種方式處理 記憶體耗盡 問題。1 判斷指標是否為null,如果是則馬上用return語句終止本函式。例如 void func void 2 判斷指標是否為null,如果是則馬上用exit...