• 技术文章 >后端开发 >php教程

    PHP FastCGI进程管理器PHP-FPM的架构

    2016-07-29 09:07:57原创954

    PHP FastCGI进程管理器PHP-FPM的架构

    一个master进程,支持多个pool,每个pool由master进程监听不同的端口,pool中有多个worker进程.
    每个worker进程都内置PHP解释器,并且进程常驻后台,支持prefork动态增加.
    每个worker进程支持在运行时编译脚本并在内存中缓存生成的opcode来提升性能.
    每个worker进程支持配置响应指定请求数后自动重启,master进程会重启挂掉的worker进程.
    每个worker进程能保持一个到MySQL/Memcached/Redis的持久连接,实现"连接池",避免重复建立连接,对程序透明.
    使用数据库持久连接时应该设置固定数量的worker进程数,不要使用动态的prefork模式.
    master进程采用epoll模型异步接收和分发请求,listen监听端口,epoll_wait等待连接,
    然后分发给对应pool里的worker进程,worker进程accpet请求后poll处理连接,
    如果worker进程不够用,master进程会prefork更多进程,
    如果prefork达到了pm.max_children上限,worker进程又全都繁忙,
    这时master进程会把请求挂起到连接队列backlog里(默认值是511).
    1个PHP-FPM工作进程在同一时刻里只能处理1个请求.
    MySQL的最大连接数max_connections默认是151.
    只要PHP-FPM工作进程数不超过151,就不会出现连接不上MySQL的情况.
    而且正常情况下,也不需要开启那么多的PHP-FPM工作进程,
    比如4个PHP-FPM进程就能跑满4个核心的CPU,
    那么你开40个PHP-FPM进程也没有任何意义,
    只会占用更多的内存,造成更多的CPU上下文切换,性能反而更差.
    为了减少每个请求都重复建立和释放连接的开销,可以开启持久连接,
    一个PHP-FPM进程保持一个到MySQL的长连接,实现透明的"连接池".
    Nginx跟PHP-FPM分开,其实是很好的解耦,PHP-FPM专门负责处理PHP请求,一个页面对应一个PHP请求,
    页面中所有静态资源的请求都由Nginx来处理,这样就实现了动静分离,而Nginx最擅长的就是处理高并发.
    PHP-FPM是一个多进程的FastCGI服务,类似Apache的prefork的进程模型,
    对于只处理PHP请求来说,这种模型是很高效很稳定的.
    不像Apache(libphp.so),一个页面,要处理多个请求,包括图片,样式表,JS脚本,PHP脚本等.
    php-fpm从5.3开始才进入PHP源代码主干,之前版本没有php-fpm.
    那时的spawn-fcgi是一个需要调用php-cgi的FastCGI进程管理器,
    另外像Apache的mod_fcgid和IIS的PHP Manager也需要调用php-cgi进程,
    但php-fpm则根本不依赖php-cgi,完全独立运行,也不依赖php(cli)命令行解释器.
    因为php-fpm是一个内置了php解释器的FastCGI服务,启动时能够自行读取php.ini配置和php-fpm.conf配置.
    个人认为,PHP-FPM工作进程数,设置为2倍CPU核心数就足够了.
    毕竟,Nginx和MySQL以及系统同样要消耗CPU.
    根据服务器内存来设置PHP-FPM进程数非常不合理,
    把内存分配给MySQL,Memcached,Redis,Linux磁盘缓存(buffers/cache)这些服务显然更合适.
    过多的PHP-FPM进程反而会增加CPU上下文切换的开销.
    PHP代码中应该尽量避免curl或者file_get_contents这些可能会产生较长网络I/O耗时的代码.
    注意设置CURLOPT_CONNECTTIMEOUT_MS超时时间,避免进程被长时间阻塞.
    如果要异步执行耗时较长的任务,可以 pclose(popen('/path/to/task.php &', 'r')); 打开一个进程来处理,
    或者借助消息队列,总之就是要尽量避免阻塞到PHP-FPM工作进程.
    在php-fpm.conf中把request_slowlog_timeout设为1秒,在slowlog中查看是否有耗时超过1秒的代码.
    优化代码,能够为所有PHP-FPM工作进程减负,这个才是提高性能的根本方法.
    能让CPU满负荷运行的操作可以视为CPU密集型操作.
    上传和下载则是典型的I/O密集型操作,因为耗时主要发生在网络I/O和磁盘I/O.
    需要PHP认证的下载操作可以委托为Nginx的AIO线程池:
    header("X-Accel-Redirect: $file_path");
    至于上传操作,比如可以建立一个监听9001端口的名为upload的PHP-FPM进程池(pool),
    专门负责处理上传操作(通过Nginx分发),避免上传操作阻塞到监听9000端口的计算密集的www进程池.
    这时upload进程池多开点进程也无所谓:
    [www]
    listen = 127.0.0.1:9000
    pm = static
    pm.max_children = 4
    [upload]
    listen = 127.0.0.1:9001
    pm = dynamic
    pm.max_children = 8
    pm.start_servers = 4
    pm.min_spare_servers = 4
    pm.max_spare_servers = 4
    利用PHP-FPM提供的池的隔离性,分离计算密集和I/O密集操作,可以减少阻塞对整个PHP应用的影响.

    文章由开源中国博客eechen原创。

    以上就介绍了PHP FastCGI进程管理器PHP-FPM的架构,包括了方面的内容,希望对PHP教程有兴趣的朋友有所帮助。

    声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn核实处理。
    专题推荐:nbsp PHP FPM worker master
    上一篇:php的solr的操作类及demo 下一篇:自己动手写 PHP MVC 框架(40节精讲/巨细/新人进阶必看)

    相关文章推荐

    • PHP网站常见一些安全漏洞及防御方法• PHP魔术方法的简单调用实例_PHP教程• 你是优秀的PHP程序员吗?_PHP教程• 实战PHP5+Apache2.2+MySQL5.1配置经历_PHP教程• PHP的内码转换函数 mb_convert_encoding()_PHP教程
    1/1

    PHP中文网