PHP无法纯代码实现MP4转换,必须依赖ffmpeg等外部工具;其本质是调用命令行,而非内置编码能力,需注意权限、超时、路径及参数适配等问题。
PHP 本身不支持直接将其他格式(比如 GIF、图片序列、文本)“转成 MP4”,它没有内置的视频编码能力。所谓“PHP 转 MP4”,实际是调用外部命令行工具(最常见的是 ffmpeg)来完成转换,PHP 只负责拼参数、执行命令、处理返回结果。
MP4 是容器格式,内部通常包含 H.264/H.265 视频流和 AAC 音频流,编码过程涉及大量数学运算与硬件加速逻辑。PHP 是通用脚本语言,不具备音视频编解码器实现。试图用纯 PHP 写 MP4 编码器,等同于重写 FFmpeg 的核心——不现实,也不安全。
file_get_contents() 读取一个 .mov 文件再 file_put_contents() 改后缀为 .mp4 → 不是转换,只是改名,播放会失败imagick 扩展处理单帧图片 → 只能生成 GIF 或 WebP,无法输出带时间轴的 MP4 视频ffmpeg 或 PATH 未配置时调用 exec('ffmpeg -version') 返回空或报错,但 PHP 不报异常,容易误判成功前提是服务器已安装 ffmpeg,且 PHP 进程有执行权限(尤其在共享主机或禁用函数环境下,exec、shell_exec 可能被禁用)。
if (function_exists('exec') && !in_array('exec', array_map('trim', explode(',', ini_get('disable_functions'))))) {
$input = '/path/to/input.avi';
$output = '/path/to/output.mp4';
$cmd = sprintf('ffmpeg -i %s -c:v libx264 -c:a aac -y %s 2>&1', escapeshellarg($input), escapeshellarg($output));
$output_log = shell_exec($cmd);
if (strpos($output_log, 'video:') !== false || file_exists($output)) {
echo "转换完成";
} else {
error_log("FFmpeg failed: " . $output_log);
}
} else {
die("exec() disabled or not available");
}
escapeshellarg() 必须使用,否则含空格或特殊字符的路径会导致命令截断或注入-y 表示自动确认覆盖,否则交互式等待输入会卡住 PHP 进程2>&1 把错误输出合并到标准输出,否则 shell_exec 拿不到报错信息system() 直接输出日志到浏览器,生产环境可能泄露路径、版本等敏感信息不同输入源需要不同处理策略,硬套同一组参数易失败或质量差。
ffmpeg -i input.gif -f mp4 -vcodec libx264 -r 15 -s 480x270 -y output.mp4
img001.jpg 格式,用 %03d 占位ffmpeg -framerate 24 -i img%03d.jpg -c:v libx264 -pix_fmt yuv420p -y out.mp4
-loop 1 循环图片,-t 控制时长ffmpeg -loop 1 -i cover.jpg -i audio.mp3 -c:v libx264 -t 180 -c:a aac -shortest -y video.mp4
视频转换是 CPU 和 I/O 密集型操作,PHP 默认设置极易导致失败。
set_time_limit(0) 必须加,否则 30 秒超时中断编码进程memory_limit 影响不大(ffmpeg 自行管理内存),但 PHP 脚本需避免缓存整个日志输出fastcgi_read_timeout / proxy_read_timeout,需同步调大
\\ 或正斜杠,且 ffmpeg.exe 路径必须绝对,相对路径常失效真正难的不是写那几行 shell_exec,而是确保 ffmpeg 版本兼容、参数组合合理、错误可捕获、资源不耗尽、用户不感知卡顿。很多“PHP 转 MP4 失败”,问题从来不在 PHP 代码里。