PHP-FPM優化及壓力測試
php fastcgi進程管理器,相比fastcgi靜態的喚起cgi,fpm能根據訪問的壓力動態的喚起cgi進程和銷燬以到達動態的調整cgi數量,這樣可以有效的使用內存。除此之外還有其它的一些優點,比如,fpm還可以平滑的重載php配置;由於fpm是使用Unix-Socket來和服務器通訊,所以也不用再配置cgi端口;fpm有更好的狀態輸出和slowlog日誌,502的時候能給出更多的錯誤細節。
本文談談有關PHP-FPM的配置細節以及基於一個實際環境來做壓力測試:
- 服務器配置如下:
- 4核心4G內存
- Nginx版本爲1.7.0
- PHP版本爲5.3.27
Nginx配置文件如下
- user www www;
- worker_processes 8;
- error_log /var/log/nginx/error.log warn;
- worker_rlimit_nofile 204800;
- pid logs/nginx.pid;
- events {
- use epoll;
- worker_connections 204800;
- }
- http {
- include mime.types;
- default_type application/octet-stream;
- log_format main '$remote_addr - $remote_user [$time_local] "$request" '
- '$status $body_bytes_sent "$http_referer" '
- '"$http_user_agent" "$http_x_forwarded_for"';
- access_log /var/log/nginx/access.log main;
- index index.shtml index.php index.html index.htm;
- server_names_hash_bucket_size 128;
- client_header_buffer_size 32k;
- open_file_cache max=204800 inactive=20s;
- ###我這邊測試,這個開啓後對php處理性能有15%提升###
- open_file_cache_valid 30s;
- open_file_cache_min_uses 1;
- large_client_header_buffers 4 32k;
- client_max_body_size 300m;
- sendfile on;
- tcp_nopush on;
- keepalive_timeout 60;
- tcp_nodelay on;
- server_tokens off;
- client_body_buffer_size 512k;
- fastcgi_cache_path /usr/local/nginx/fastcgi_cache levels=1:2 keys_zone=TEST:10m inactive=5m;
- fastcgi_connect_timeout 300;
- fastcgi_send_timeout 300;
- fastcgi_read_timeout 300;
- fastcgi_buffer_size 64k;
- fastcgi_buffers 4 64k;
- fastcgi_busy_buffers_size 128k;
- fastcgi_temp_file_write_size 128k;
- fastcgi_cache TEST;
- ###這個開啓後感覺效果非常明顯,有效的降低cpu負載,提高了php的處理能力###
- fastcgi_cache_use_stale error timeout invalid_header http_500;
- fastcgi_intercept_errors on;
- fastcgi_cache_valid 200 302 1h;
- fastcgi_cache_valid 301 1d;
- fastcgi_cache_valid any 1m;
- fastcgi_cache_min_uses 1;
- gzip on;
- gzip_min_length 1k;
- gzip_buffers 4 16k;
- gzip_http_version 1.1;
- gzip_comp_level 2;
- gzip_vary on;
- include /usr/local/nginx/conf.d/*.conf;
- ssi on;
- ssi_silent_errors on;
- ssi_types text/shtml;
- upstream phpbackend {
- server unix:/dev/shm/php-fpm.sock weight=100 max_fails=10 fail_timeout=30;
- server unix:/dev/shm/php-fpm2.sock weight=100 max_fails=10 fail_timeout=30;
- server unix:/dev/shm/php-fpm3.sock weight=100 max_fails=10 fail_timeout=30;
- }
- server {
- listen 80;
- server_name 10.168.41.2 default backlog=204800;
- location / {
- root html;
- index index.html index.htm;
- }
- error_page 500 502 503 504 /50x.html;
- location = /50x.html {
- root html;
- }
- location ~ \.php$ {
- fastcgi_pass phpbackend;
- fastcgi_index index.php;
- fastcgi_param SCRIPT_FILENAME /usr/local/nginx/html$fastcgi_script_name;
- include fastcgi_params;
- }
- }
- }
優化後的php-fpm配置如下:我開啓了3個php-fpm實例,每個實例爲60個進程,使用socket連接FastCGI,這個可以根據自己的服務器配置進行調整
- listen = /dev/shm/php-fpm.sock
- ###/dev/shm是內存文件系統,放在內存中肯定會快###
- pm.start_servers = 60
- pm.min_spare_servers = 40
- pm.max_spare_servers = 80
- ###Default Value: min_spare_servers + (max_spare_servers - min_spare_servers) / 2###
- pm.max_requests = 204800
- rlimit_files = 102400
- listen.backlog = 102400
- ###隊列等待長度,如果出現11: Resource temporarily unavailable錯誤,可以把這個值調高一點,分別對應系統內核backlog,nginx的backlog###
內核參數優化如下
- net.ipv4.tcp_max_tw_buckets = 250000
- ###同時保持TIME_WAIT套接字的最大數量,如果超過這個數字,TIME_WAIT套接字將立刻被清除並打印警告信息###
- net.ipv4.tcp_sack = 1
- ###通過有選擇地應答亂序接收到的報文來提高性能,讓發送者只發送丟失的報文段,(對於廣域網通信來說)這個選項應該啓用,但是會增加對CPU的佔用###
- net.ipv4.tcp_window_scaling = 1
- ###支持更大的TCP窗口 如果TCP窗口最大超過65535(64K)必須設置該數值爲1###
- net.ipv4.tcp_rmem = 4096 87380 4194304
- ###這個參數定義了TCP接收緩存(用於TCP接收滑動窗口)的最小值、默認值、最大值###
- net.ipv4.tcp_wmem = 4096 16384 4194304
- ###這個參數定義了TCP發送緩存(用於TCP發送滑動窗口)的最小值、默認值、最大值###
- net.core.wmem_default = 8388608
- ###這個參數表示內核套接字發送緩存區默認的大小###
- net.core.rmem_default = 8388608
- ###這個參數表示內核套接字接收緩存區默認的大小###
- net.core.rmem_max = 16777216
- ###這個參數表示內核套接字接收緩存區的最大大小###
- net.core.wmem_max = 16777216
- ###這個參數表示內核套接字發送緩存區的最大大小###
- net.core.netdev_max_backlog = 262144
- ###當網卡接收數據包的速度大於內核處理的速度時,會有一個隊列保存這些數據包,這個參數表示該隊列的最大值###
- net.core.somaxconn = 262144
- ###socket監聽(listen)的backlog上限,在高併發的請求中可能會導致鏈接超時或者觸發重傳###
- net.ipv4.tcp_max_orphans = 3276800
- ###設定系統中最多有多少個TCP套接字不被關聯到任何一個用戶文件句柄上,超過了會print to message###
- net.ipv4.tcp_max_syn_backlog = 262144
- ###這個參數表示TCP三次握手建立階段接收SYN請求隊列的最大長度,默認是1024,將其設置的大一些可以使出現Nginx繁忙來不及Accept新連接的情況時,Linux不至於丟失客戶端發起的連接請求###
- net.ipv4.tcp_timestamps = 0
- ###開啓對於TCP時間戳的支持,其時間戳在(請參考RFC 1323)TCP的包頭增加12個字節###
- net.ipv4.tcp_synack_retries = 1
- ###SYN-ACK握手狀態重試次數,默認爲5,遭受syn-flood攻擊時改爲1或2###
- net.ipv4.tcp_syn_retries = 1
- ###外向syn握手重試次數,默認爲4###
- net.ipv4.tcp_tw_recycle = 1
- ###表示開啓TCP連接中TIME-WAIT sockets的快速回收,默認爲0,表示關閉###
- net.ipv4.tcp_tw_reuse = 1
- ###表示開啓重用。允許將TIME-WAIT sockets重新用於新的TCP連接,默認爲0,表示關閉###
- net.ipv4.tcp_mem = 94500000 915000000 927000000
- ###三個文本字段,超過第三個的字段TCP會拒絕SOCKET,其單位是內存頁###
- net.ipv4.tcp_fin_timeout = 1
- ###這個參數表示當服務器主動關閉連接時,socket保持在FIN-WAIT-2狀態的最大時間###
- net.ipv4.tcp_keepalive_time = 30
- ###當keepalive啓用時,TCP發送keepalive消息的頻度,默認是2小時,若將其設置的小一些,可以更快的清理無效的連接###
- net.ipv4.ip_local_port_range = 1024 65000
- ###這個參數定義了在UDP和TCP連接中本地(不包括連接的遠端)端口取值範圍###
- net.netfilter.nf_conntrack_tcp_timeout_established = 36000
- ###established的超時時間###
- net.nf_conntrack_max = 655360
- ###這個值決定了你作爲網關的工作能力上限,默認65536###
- net.ipv4.tcp_syncookies = 0
- ###關閉SYN Cookies,默認關閉###
調高linux內核打開文件數量
- echo ‘ulimit -HSn 102400′ >> /etc/profile
- echo ‘ulimit -HSn 102400′ >> /etc/rc.local
- source /etc/profile
使用ab壓力測試(頁面爲純php頁面,未讀取數據庫)
開啓緩存後的效果:1W3的ops,cpu負載爲20%
關閉緩存後的效果:ops爲:9700,cpu負載爲100%