让我们来迅速了解一下怎样设置 PHP-FPM,以便达到高吞吐,低延迟以及稳定的使用 CPU 和内存的完美状态。在默认的情况下,大多数设置都将 PHP-FPM PM(进程管理器)设置为
dynamic
,或者当你有可用内存的问题时常建议你使用 ondemand
。接下来,让我们根据 php.net 的官方文档来比较一下这两个管理选项和我最常用的设置 —— static
之间的区别:
pm = dynamic:子进程的数量是根据以下指令来动态生成的:
pm.max_children
,pm.start_servers
,pm.min_spare_servers
,pm.max_spare_servers
.pm = ondemand:在服务启动的时候根据
pm.start_servers
指令生成进程,而非动态生成。pm = static:子进程的数量是由
pm.max_children
指令来确定的。
查看完整列表,深入了解 php-fpm.conf
的所有指令。
PHP-FPM 进程管理器(PM)和 CPUFreq Governor 的相似之处
现在,我们要说些偏离主题,但我觉得和 PHP-FPM 调优有关的事情。好了,我们都有过在某些时候的 CPU 缓慢问题,无论是笔记本电脑、VM 或者是专用的服务器。还记得 CPU 频率缩放问题吗?(CPUFreq governor)这些设置在类 Unix 系统和 Windows 上是有效的,可以通过修改 CPU governor,将其从 ondemand
修改为 performance
来提高性能并加快系统的响应。现在,让我们来比较下列 CPUFreq governor 描述和 PHP-FPM PM 有哪些相似之处:
Governor = ondemand:根据当前负荷动态调整 CPU 频率。先将 CPU 频率调整至最大,然后随着空闲时间的增加而缩小频率。
Governor = conservative:根据当前负荷动态调整频率。比设置成
ondemand
更加缓慢。Governor = performance:始终以最大频率运行 CPU。
查看 CPUFreq governor 选项详细列表 ,获取更多相关信息。
注意到相似之处了吗?这就是我这个比较的首要目的,为了找到一个最好的方式来写这篇文章,推荐你将 PHP-FPM 的 pm static
当作你的第一选择。
使用 CPU Governor 的 performance
设置是一个非常安全的性能提升方式,因为它能完美的使用你服务器 CPU 的全部性能。唯一需要考虑的因素就是一些诸如散热、电池寿命(笔记本电脑)和一些由 CPU 始终保持 100% 所带来的一些副作用。一旦设置为 performance
,那么它确实是你 CPU 最快的设置。相关实例请阅读 'force_turbo' 在 Raspberry Pi 上的设置,它教你在 RPi 板上使用 performance
Governor,由于 CPU 时钟速度较低,性能改善将更加明显。
使用 pm static
优化你的服务器性能
PHP-FPM 的 static
设置取决于你服务器有多少闲置内存。大多数情况下,如果你服务器的内存不足,那么 PM 设置成 ondemand
或 dynamic
将是更好的选择。但是,一旦你有可用的闲置内存,那么把 PM 设置成 static
的最大值将减少许多 PHP 进程管理器(PM)所带来的开销。换句话说,你应该在没有内存不足和缓存压力的情况下使用 pm.static
来设置 PHP-FPM 进程的最大数量。此外,也不能影响到 CUP 的使用和其他待处理的 PHP-FPM 操作。
在上面的截图中,这台服务器的设置( pm = static
,pm.max_children =100
)最多使用了 10GB 的内存。请注意高亮的列。Google 分析图中大概有 200 个活跃用户(60秒内)。在这种用户量下,有 70% 的 PHP-FPM 子进程被闲置。这意味着,无论当前流量如何,PHP-FPM 始终保持着足够多的进程。闲置的进程始终保持在线,就算达到了流量的峰值也能快速响应,而不是等待 PM 生成子进程,然后在 x pm.process_idle_timeout
秒后将此进程结束。我将 pm.max_requests
设置的非常高,因为这是一个不可能发生内存泄漏的 PHP 生产服务器。如果你对你的 PHP 脚本有着 110% 的信心,那么你可用选择使用 pm.max_requests = 0
。但建议适当的重启服务。将请求数量设置的很高,是为了避免过高的 PM 开销。例如,设置 pm.max_requests = 1000
,但这需要根据 pm.max_children
的设置和实际每秒的请求数量来决定。
截图使用 Linux top 通过 'u'(user)选项和 PHP-FPM 用户名进行过滤。并只显示了前 50 个左右(未统计)的进程,但基本上 top
命令也只会显示适合你终端窗口大小的内容 —— 在本例中,使用 %CPU
排序。要查看全部的 100 条 PHP-FPM 进程的话,你需要使用以下命令:
top -bn1 | grep php-fpm
何时使用 ondemand 和 dynamic
使用 pm dynamic
,您可能会出现类似于下面的错误:
WARNING: [pool xxxx] seems busy (you may need to increase pm.start_servers, or pm.min/max_spare_servers), spawning 32 children, there are 4 idle, and 59 total children
您可能会尝试调整 pm 配置,但仍然会看到同样的错误
在这种情况下,pm.min
太低,并且因为流量和峰值波动很大,使用 pm dynamic
可能难以调整
一般的建议是使用 pm ondemand
。 然而,情况会变的更糟,因为 ondemand
会在没有流量时关闭空闲进程,然后最终会产生与流量波动很大一样的开销问题 (除非您设置空闲超时的时间非常非常的长)
但是,当您拥有多个 pm 进程池时,pm dynamic
, 特别是 ondemand
是可以为您节省时间的。例如在共享的 VPS 上,有 100+ 的 cPanel 账号和 200+ 的域名,使用 pm.static
或者是 pm.dynamic
都是不可能的,即使在没有任何流量的情况下,内存会被瞬间用完,而 pm.ondemand
意味着所有空闲的子进程都会被完全关闭,节省了大量内存。cPanel 的开发者已经意识到了这个问题,现在的 cPanel 默认就是设置为 pm.ondemand
。
结论
--
当流量波动比较大的时候,,PHP-FPM 的 ondemand
和 dynamic
会因为固有开销而限制吞吐量。 您需要了解您的系统并设置 PHP-FPM 进程数,以匹配服务器的最大容量。
从 pm.max_children
开始,根据 pm dynamic
或 ondemand
的最大使用情况去设置
您会注意到,在 pm static
模式下,因为您将所有内容都保存在内存中,所以随着时间的推移,流量峰值会对 CPU 造成比较小的峰值,并且您的服务器负载和 CPU 平均值将变得更加平滑。 每个需要手动调整的 PHP-FPM 进程数的平均大小会有所不同
更新:附上一张 A/B 测试图。
转自 PHP / Laravel 开发者社区 https://laravel-china.org/topics/14952