PHP与Web服务器兼容性取决于服务器类型、运行模式(mod_php/FPM/CGI)及PHP编译方式是否匹配:Apache 2.4需mod_php且SAPI为apache2handler;Nginx必须搭配php-fpm(SAPI为fpm-fcgi)并正确配置fastcgi_pass;Windows IIS需FastCGI且fastcgi.impersonate=1。
PHP 和 Web 服务器不是随便装就能跑起来的,尤其在生产环境或升级时,版本不匹配会直接导致 500 Internal Server Error、Module not found 或 PHP 文件被直接下载(未解析)。关键看三点:Web 服务器类型、其运行模式(mod_php / FPM / CGI),以及 PHP 的编译方式是否匹配。
mod_php:PHP 必须以 Apache 模块方式编译,且 php_version 与 apache2handler SAPI 匹配;PHP 8.0+ 已移除对 Apache 2.2 的支持php-fpm:PHP 必须启用 fpm SAPI(即编译时含 --enable-fpm),Nginx 本身不解析 PHP,只反向代理到 php-fpm 监听地址(如 127.0.0.1:9000 或 /run/php/php8.2-fpm.sock)FastCGI:需确保 php-cgi.exe 路径正确,且 php.ini 中 fastcgi.impersonate = 1 开启(否则权限异常)光看 php -v 不够——它只告诉你 CLI 版本。真正决定 Web 环境行为的是 SAPI 类型和加载方式。用 phpinfo() 最可靠,但命令行下可快速验证:
php -r "echo php_sapi_name();"
输出常见值含义:
apache2handler → PHP 作为 Apache 模块加载,需检查 LoadModule php_module 行是否存在于 Apache 配置中fpm-fcgi → 正在运行 php-fpm,Nginx/Apache 必须配置 FastCGI 代理cli → 当前只是命令行环境,Web 请求根本不会走这个进程同时确认 Web 服务器实际加载的 PHP 配置路径:
php -r "echo php_ini_loaded_file();"
这个 php.ini 文件必须和 Web 进程读取的一致;若不一致(比如 CLI 用一个,FPM 用另一个),extension_dir、date.timezone 等设置就可能失效。
这些现象背后基本都是版本或 SAPI 错配:
Cannot load modules/libphp.so into server: ... undefined symbol: core_globals → PHP 编译版本与 Apache ABI 不兼容(如 PHP 8.2 用 Apache 2.2 编译)502 Bad Gateway,php-fpm 日志空或报 WARNING: [pool www] child 1234 exited on signal 11 (SIGSEGV) → PHP FPM 二进制与系统 glibc 版本冲突(常见于 Alpine 容器里混用非 musl 编译的 PHP).php 请求转给 PHP 处理,检查 location ~ \.php$ 是否遗漏或 fastcgi_pass 地址写错libphp.so,但 phpinfo() 显示 SAPI 是 cli → Apache 实际没生效该模块(可能是 httpd.conf 里没 Include 对应 conf,或 SELinux 阻止了模块加载)PHP 小版本升级(如 8.1.22 → 8.1.23)通常不影响 Web 集成,但大版本(7.4 → 8.0)或操作系统级更新(Ubuntu 20.04 → 22.04)很可能破坏原有搭配。验证步骤要闭环:
sudo systemctl restart apache2 或 sudo systemctl restart php8.2-fpm nginx
ps aux | grep php-fpm 或 apache2ctl -M | grep php
test.php(内容为 ),确认页面正常且顶部显示的 Server API 字段与
Loaded Configuration File 和 Scan this dir for additional .ini files 路径,确保你改过的扩展(如 opcache、mysqli)确实在其中被加载最容易被忽略的是:不同 PHP 版本共存时,php-fpm 的 socket 文件名或端口常被硬编码在 Nginx 配置里(如 fastcgi_pass unix:/run/php/php7.4-fpm.sock),升级到 8.2 后忘了同步改,结果流量全打到一个不存在的 socket 上。