Nginx使用的php-fpm的两种进程管理方式及优化

php-fpm目前主要又两个分支,分别对应于php-5.2.x的版本和php-5.3.x的版本。在5.2.x的版本中,php-fpm.conf使用的是xml格式,而在新的5.3.x版本中,则是和php.ini一样的配置风格。

对于进程的管理存在两种风格——static和dynamic。和之前的版本的进程管理其实还是一样的,只是将apache-like改成了dynamic,这样更容易理解。

如果设置成static,php-fpm进程数自始至终都是pm.max_children指定的数量,不再增加或减少。

如果设置成dynamic,则php-fpm进程数是动态的,最开始是pm.start_servers指定的数量,如果请求较多,则会自动增加,保证空闲的进程数不小于pm.min_spare_servers,如果进程数较多,也会进行相应清理,保证多余的进程数不多于pm.max_spare_servers。

这两种不同的进程管理方式,可以根据服务器的实际需求来进行调整。

这里先说一下涉及到这个的几个参数,他们分别是pmpm.max_childrenpm.start_serverspm.min_spare_serverspm.max_spare_servers

pm表示使用那种方式,有两个值可以选择,就是static(静态)或者dynamic(动态)。在更老一些的版本中,dynamic被称作apache-like。这个要注意看配置文件的说明。

下面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程序在执行完成后,或多或少会有内存泄露的问题。这也是为什么开始的时候一个php-fpm进程只占用3M左右内存,运行一段时间后就会上升到20-30M的原因了。

对于内存大的服务器(比如8G以上)来说,指定静态的max_children实际上更为妥当,因为这样不需要进行额外的进程数目控制,会提高效率。因为频繁开关php-fpm进程也会有时滞,所以内存够大的情况下开静态效果会更好。数量也可以根据 内存/30M 得到,比如8GB内存可以设置为100,那么php-fpm耗费的内存就能控制在 2G-3G的样子。如果内存稍微小点,比如1G,那么指定静态的进程数量更加有利于服务器的稳定。这样可以保证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.max_requests = 5000
//每个子进程重生之前服务的请求数,主要目的是为了控制请求处理过程中的内存溢出,使得内存占用在一个可接受的范围内。如果设置为 ‘0’ 则一直接受请求。它不能设置为非常小的值,当这个数字非常小的时候,由于PHP请求是平均地分配给各个工作进程的,如果这个值太小就会导致所有的工作进程几乎同时达到这个值并且进入该进程自身重启的状态,当所有的工作进程都在同一时刻重启就会发生在数秒内甚至更长的时间PHP将停止响应直到所有的进程均重启完为止,此时有新请求时则无法被正常处理,只能排队待进程重启完成,如果超过了超时时间,会提示502 bad gateway。同时因为进程频繁停止与创建,可以看到CPU利用率有明显的增加。所以这个值需要调整稍微大一些。这个值可以通过以下的方式进行估算:
pm.max_requests = 进程重启间隔时间*并发量/pm.max_children
举例:
所有子进程重启时间为300s一次,并发量为50,max_children设置为25,max_requests = 300*50/25=600,如果想让进程每隔一小时重生一次,则是3600*50/25=7200。

查看进程:

ps -ylC php-fpm --sort:rss

喜欢的话订阅一个呗~第一时间收到文章更新哟~

5条回应:“Nginx使用的php-fpm的两种进程管理方式及优化”

  1. 匿名说道:

    学习

  2. 匿名说道:

    请问,“所有子进程重启时间”指的是什么?能解释一下吗?

    • 天堂说道:

      指的是进程最大的运行时间,这个这nginx配置网站的时候可以设置,默认300s。比如一个用户访问你的网站开启一个php-fpm进程,如果他300s始终未结束(死循环),那么nginx会杀掉他,一般是几秒就结束了,结束之后nginx会将他放在静默等待池。

      • 匿名说道:

        具体的参数是什么? 是php.ini中的max_execution_time吗?

        • 天堂说道:

          是的,他依赖于php-fpm的运行。参考下这个,有三个地方要改(加):https://stackoverflow.com/questions/22658908/set-ini-max-execution-time-doesnt-work

发表评论

电子邮件地址不会被公开。