php fastcgi进程管理器,相比fastcgi静态的唤起cgi,fpm能根据访问的压力动态的唤起cgi进程和销毁以到达动态的调整cgi数量,这样可以有效的使用内存。除此之外还有其它的一些优点,比如,fpm还可以平滑的重载php配置;由于fpm是使用Unix-Socket来和服务器通讯,所以也不用再配置cgi端口;fpm有更好的状态输出和slowlog日志,502的时候能给出更多的错误细节。

 

本文谈谈有关PHP-FPM的配置细节以及基于一个实际环境来做压力测试:

  1. 服务器配置如下:
  2. 4核心4G内存
  3. Nginx版本为1.7.0
  4. PHP版本为5.3.27

 

Nginx配置文件如下

  1. user www www;
  2. worker_processes 8;
  3. error_log /var/log/nginx/error.log warn;
  4. worker_rlimit_nofile 204800;
  5. pid logs/nginx.pid;
  6. events {
  7. use epoll;
  8. worker_connections 204800;
  9. }
  10. http {
  11. include mime.types;
  12. default_type application/octet-stream;
  13. log_format main '$remote_addr - $remote_user [$time_local] "$request" '
  14. '$status $body_bytes_sent "$http_referer" '
  15. '"$http_user_agent" "$http_x_forwarded_for"';
  16. access_log /var/log/nginx/access.log main;
  17. index index.shtml index.php index.html index.htm;
  18. server_names_hash_bucket_size 128;
  19. client_header_buffer_size 32k;
  20. open_file_cache max=204800 inactive=20s;
  21. ###我这边测试,这个开启后对php处理性能有15%提升###
  22. open_file_cache_valid 30s;
  23. open_file_cache_min_uses 1;
  24. large_client_header_buffers 4 32k;
  25. client_max_body_size 300m;
  26. sendfile on;
  27. tcp_nopush on;
  28. keepalive_timeout 60;
  29. tcp_nodelay on;
  30. server_tokens off;
  31. client_body_buffer_size 512k;
  32. fastcgi_cache_path /usr/local/nginx/fastcgi_cache levels=1:2 keys_zone=TEST:10m inactive=5m;
  33. fastcgi_connect_timeout 300;
  34. fastcgi_send_timeout 300;
  35. fastcgi_read_timeout 300;
  36. fastcgi_buffer_size 64k;
  37. fastcgi_buffers 4 64k;
  38. fastcgi_busy_buffers_size 128k;
  39. fastcgi_temp_file_write_size 128k;
  40. fastcgi_cache TEST;
  41. ###这个开启后感觉效果非常明显,有效的降低cpu负载,提高了php的处理能力###
  42. fastcgi_cache_use_stale error timeout invalid_header http_500;
  43. fastcgi_intercept_errors on;
  44. fastcgi_cache_valid 200 302 1h;
  45. fastcgi_cache_valid 301 1d;
  46. fastcgi_cache_valid any 1m;
  47. fastcgi_cache_min_uses 1;
  48. gzip on;
  49. gzip_min_length 1k;
  50. gzip_buffers 4 16k;
  51. gzip_http_version 1.1;
  52. gzip_comp_level 2;
  53. gzip_vary on;
  54. include /usr/local/nginx/conf.d/*.conf;
  55. ssi on;
  56. ssi_silent_errors on;
  57. ssi_types text/shtml;
  58. upstream phpbackend {
  59. server unix:/dev/shm/php-fpm.sock weight=100 max_fails=10 fail_timeout=30;
  60. server unix:/dev/shm/php-fpm2.sock weight=100 max_fails=10 fail_timeout=30;
  61. server unix:/dev/shm/php-fpm3.sock weight=100 max_fails=10 fail_timeout=30;
  62. }
  63. server {
  64. listen 80;
  65. server_name 10.168.41.2 default backlog=204800;
  66. location / {
  67. root html;
  68. index index.html index.htm;
  69. }
  70. error_page 500 502 503 504 /50x.html;
  71. location = /50x.html {
  72. root html;
  73. }
  74. location ~ \.php$ {
  75. fastcgi_pass phpbackend;
  76. fastcgi_index index.php;
  77. fastcgi_param SCRIPT_FILENAME /usr/local/nginx/html$fastcgi_script_name;
  78. include fastcgi_params;
  79. }
  80. }
  81. }

 

优化后的php-fpm配置如下:我开启了3个php-fpm实例,每个实例为60个进程,使用socket连接FastCGI,这个可以根据自己的服务器配置进行调整

  1. listen = /dev/shm/php-fpm.sock
  2. ###/dev/shm是内存文件系统,放在内存中肯定会快###
  3. pm.start_servers = 60
  4. pm.min_spare_servers = 40
  5. pm.max_spare_servers = 80
  6. ###Default Value: min_spare_servers + (max_spare_servers - min_spare_servers) / 2###
  7. pm.max_requests = 204800
  8. rlimit_files = 102400
  9. listen.backlog = 102400
  10. ###队列等待长度,如果出现11: Resource temporarily unavailable错误,可以把这个值调高一点,分别对应系统内核backlog,nginx的backlog###

 

内核参数优化如下

  1. net.ipv4.tcp_max_tw_buckets = 250000
  2. ###同时保持TIME_WAIT套接字的最大数量,如果超过这个数字,TIME_WAIT套接字将立刻被清除并打印警告信息###
  3. net.ipv4.tcp_sack = 1
  4. ###通过有选择地应答乱序接收到的报文来提高性能,让发送者只发送丢失的报文段,(对于广域网通信来说)这个选项应该启用,但是会增加对CPU的占用###
  5. net.ipv4.tcp_window_scaling = 1
  6. ###支持更大的TCP窗口 如果TCP窗口最大超过65535(64K)必须设置该数值为1###
  7. net.ipv4.tcp_rmem = 4096 87380 4194304
  8. ###这个参数定义了TCP接收缓存(用于TCP接收滑动窗口)的最小值、默认值、最大值###
  9. net.ipv4.tcp_wmem = 4096 16384 4194304
  10. ###这个参数定义了TCP发送缓存(用于TCP发送滑动窗口)的最小值、默认值、最大值###
  11. net.core.wmem_default = 8388608
  12. ###这个参数表示内核套接字发送缓存区默认的大小###
  13. net.core.rmem_default = 8388608
  14. ###这个参数表示内核套接字接收缓存区默认的大小###
  15. net.core.rmem_max = 16777216
  16. ###这个参数表示内核套接字接收缓存区的最大大小###
  17. net.core.wmem_max = 16777216
  18. ###这个参数表示内核套接字发送缓存区的最大大小###
  19. net.core.netdev_max_backlog = 262144
  20. ###当网卡接收数据包的速度大于内核处理的速度时,会有一个队列保存这些数据包,这个参数表示该队列的最大值###
  21. net.core.somaxconn = 262144
  22. ###socket监听(listen)的backlog上限,在高并发的请求中可能会导致链接超时或者触发重传###
  23. net.ipv4.tcp_max_orphans = 3276800
  24. ###设定系统中最多有多少个TCP套接字不被关联到任何一个用户文件句柄上,超过了会print to message###
  25. net.ipv4.tcp_max_syn_backlog = 262144
  26. ###这个参数表示TCP三次握手建立阶段接收SYN请求队列的最大长度,默认是1024,将其设置的大一些可以使出现Nginx繁忙来不及Accept新连接的情况时,Linux不至于丢失客户端发起的连接请求###
  27. net.ipv4.tcp_timestamps = 0
  28. ###开启对于TCP时间戳的支持,其时间戳在(请参考RFC 1323)TCP的包头增加12个字节###
  29. net.ipv4.tcp_synack_retries = 1
  30. ###SYN-ACK握手状态重试次数,默认为5,遭受syn-flood攻击时改为1或2###
  31. net.ipv4.tcp_syn_retries = 1
  32. ###外向syn握手重试次数,默认为4###
  33. net.ipv4.tcp_tw_recycle = 1
  34. ###表示开启TCP连接中TIME-WAIT sockets的快速回收,默认为0,表示关闭###
  35. net.ipv4.tcp_tw_reuse = 1
  36. ###表示开启重用。允许将TIME-WAIT sockets重新用于新的TCP连接,默认为0,表示关闭###
  37. net.ipv4.tcp_mem = 94500000 915000000 927000000
  38. ###三个文本字段,超过第三个的字段TCP会拒绝SOCKET,其单位是内存页###
  39. net.ipv4.tcp_fin_timeout = 1
  40. ###这个参数表示当服务器主动关闭连接时,socket保持在FIN-WAIT-2状态的最大时间###
  41. net.ipv4.tcp_keepalive_time = 30
  42. ###当keepalive启用时,TCP发送keepalive消息的频度,默认是2小时,若将其设置的小一些,可以更快的清理无效的连接###
  43. net.ipv4.ip_local_port_range = 1024 65000
  44. ###这个参数定义了在UDP和TCP连接中本地(不包括连接的远端)端口取值范围###
  45. net.netfilter.nf_conntrack_tcp_timeout_established = 36000
  46. ###established的超时时间###
  47. net.nf_conntrack_max = 655360
  48. ###这个值决定了你作为网关的工作能力上限,默认65536###
  49. net.ipv4.tcp_syncookies = 0
  50. ###关闭SYN Cookies,默认关闭###

 

调高linux内核打开文件数量

  1. echo ulimit -HSn 102400 >> /etc/profile
  2. echo ulimit -HSn 102400 >> /etc/rc.local
  3. source /etc/profile

 

使用ab压力测试(页面为纯php页面,未读取数据库)

开启缓存后的效果:1W3的ops,cpu负载为20%

 

php-fpmtesting

php-fpmtesting

 

php-fpmtesting1

php-fpmtesting1

 

关闭缓存后的效果:ops为:9700,cpu负载为100%

 

php-fpmtesting2

php-fpmtesting2

 

php-fpmtesting3

php-fpmtesting3