dynamic模式需在pool配置中设置pm=dynamic并配全5个关键参数:pm.max_children、pm.start_servers、pm.min_spare_servers、pm.max_spare_servers、pm.max_requests;修改后须reload服务且验证运行时状态。

php-fpm.conf 里怎么配 dynamic 模式
动态进程池对应的是 pm = dynamic,不是默认的 static 或 ondemand。它让 PHP-FPM 根据请求压力自动伸缩子进程数,适合流量波动大的场景。
关键参数必须一起配全,漏一个就可能退化成 static 行为或频繁重启:
-
pm.max_children:全局最大允许的子进程数,内存超限前的硬上限 -
pm.start_servers:FPM 启动时直接 fork 的子进程数,建议设为pm.min_spare_servers和pm.max_spare_servers的中间值 -
pm.min_spare_servers:空闲进程下限,低于此数就 fork 新进程 -
pm.max_spare_servers:空闲进程上限,高于此数就 kill 掉多余进程 -
pm.max_requests(可选但推荐):每个子进程处理多少请求后自动退出,防内存泄漏
pm = dynamic pm.max_children = 50 pm.start_servers = 10 pm.min_spare_servers = 5 pm.max_spare_servers = 20 pm.max_requests = 500
为什么改了配置没生效
常见原因不是配置写错,而是没触发重载或被覆盖:
- 改的是主
php-fpm.conf,但实际加载的是www.conf里的 pool 配置 —— 动态模式必须在 pool 级别(如/etc/php/8.1/fpm/pool.d/www.conf)里设置pm = dynamic - 改完没执行
sudo systemctl reload php8.1-fpm(Ubuntu/Debian)或sudo systemctl reload php-fpm(CentOS/RHEL),restart也行,但reload更安全 - 存在多个 pool,只改了其中一个,而 Nginx upstream 指向的是另一个未修改的 pool
-
配置文件语法错误导致 reload 失败,用
sudo php-fpm8.1 -t检查配置有效性
如何验证当前是 dynamic 模式且在正常伸缩
不能只看配置文件,得看运行时状态和日志:
立即学习“PHP免费学习笔记(深入)”;
- 查当前活跃进程数:
sudo systemctl status php8.1-fpm | grep "active (running)"然后ps aux | grep "php-fpm: pool www"数一下 worker 进程数,观察高并发时是否明显增多 - 开慢日志 +
pm.status_path:在 pool 配置中加pm.status_path = /status,再配 Nginx location 透传,访问/status?full就能看到start time、idle processes、active processes等实时数据 - 看错误日志:
/var/log/php8.1-fpm.log中出现[WARNING] [pool www] server reached pm.max_children setting,说明已触顶,需要调高pm.max_children或优化脚本
dynamic 模式下容易被忽略的内存问题
很多人以为 dynamic 就是“省资源”,其实恰恰相反:它靠多进程应对峰值,内存占用比 static 更不可控。
- 每个 PHP 子进程常驻内存约 20–60 MB(取决于扩展和 autoload),
pm.max_children = 50意味着最多可能吃掉 3 GB 内存 -
pm.max_spare_servers设太高(比如 30),低峰期仍留一堆 idle 进程占内存;设太低(比如 2),高峰期 fork 不及,请求排队 - 没配
pm.max_requests时,长期运行的进程可能因 Composer autoloader 或静态变量累积内存,最终 OOM 被系统 kill - 建议用
pm = ondemand替代 dynamic 的唯一合理场景:极低频小站(日均请求
动态池不是设完就万事大吉,它把调优压力从“固定数量”转成了“响应曲线+内存水位”的双维度监控。不看 pm.status_path 输出和系统 free -h,等于闭眼开车。











