PHP输出MP4必须设置Content-Type: video/mp4、Accept-Ranges: bytes并正确处理Range请求,否则播放器因MIME不匹配或无分段支持而拒播;需用Nginx重写伪装路径,禁用超时与中断,并避免内存溢出。

PHP 文件后缀不能“变成” MP4,浏览器和播放器只认真实 MIME 类型
直接把 .php 文件重命名为 .mp4,播放器依然会失败——因为 HTTP 响应头里的 Content-Type 是 text/html 或 application/x-httpd-php,不是 video/mp4。播放器根本不看文件名,只信响应头和实际字节流结构。
用 PHP 输出 MP4 文件时必须设置正确的 Header
常见错误是只用 readfile() 输出文件,却漏掉关键响应头。MP4 播放(尤其带拖拽、快进)依赖 Accept-Ranges: bytes 和正确 Content-Type,否则 iOS Safari、VLC、部分 H5 播放器直接拒播或卡死。
- 必须设置:
header('Content-Type: video/mp4'); - 必须设置:
header('Accept-Ranges: bytes'); - 若支持拖拽,还需处理
Range请求头(用$_SERVER['HTTP_RANGE']解析并输出分段) - 禁用缓存干扰:
header('Cache-Control: public, max-age=31536000');
header('Content-Type: video/mp4');
header('Accept-Ranges: bytes');
header('Cache-Control: public, max-age=31536000');
$filepath = '/path/to/video.mp4';
if (isset($_SERVER['HTTP_RANGE'])) {
$range = $_SERVER['HTTP_RANGE'];
// 简化 Range 解析逻辑(生产环境需完整校验)
if (preg_match('/bytes=(\d+)-(\d+)?/', $range, $matches)) {
$start = (int)$matches[1];
$end = isset($matches[2]) ? (int)$matches[2] : filesize($filepath) - 1;
$length = $end - $start + 1;
header("HTTP/1.1 206 Partial Content");
header("Content-Range: bytes $start-$end/" . filesize($filepath));
header("Content-Length: $length");
$fp = fopen($filepath, 'rb');
fseek($fp, $start);
fpassthru($fp);
fclose($fp);
exit;
}}
// 全量输出
header('Content-Length: ' . filesize($filepath));
readfile($filepath);
不要用 URL 伪装(如 video.php?id=123),而要用真实路径映射
很多播放器(尤其是移动端)会预检 URL 后缀或拒绝非静态路径。即使 PHP 输出了正确 Header,https://site.com/play.php?v=abc 在某些 H5 播放器里仍被拦截或静音。
- 推荐方案:用 Nginx/Apache 的重写规则,把
/videos/xxx.mp4映射到 PHP 处理脚本,但对外保持.mp4路径 - Nginx 示例:
location ~ ^/videos/.+\.mp4$ { rewrite ^/videos/(.+\.mp4)$ /video_proxy.php?file=$1 last; } - 这样前端
就能正常加载,且支持拖拽、HLS 兼容、CDN 缓存
PHP 输出 MP4 的性能和内存风险必须处理
直接 readfile() 大文件易触发内存溢出或超时;不设 set_time_limit(0) 和 ignore_user_abort(true) 可能中途断连导致视频残缺。
立即学习“PHP免费学习笔记(深入)”;
- 务必加:
set_time_limit(0);(禁用超时) - 加:
ignore_user_abort(true);(用户关页面也不中断输出) - 大文件建议用
fopen + fpassthru替代file_get_contents,避免内存吃满 - 如果文件在私有目录(如
/var/www/private/videos/),确保 PHP 进程有读权限,且路径不被 Web 服务器直接暴露
实际最难的不是改后缀,而是让 PHP 输出行为完全模拟静态 MP4 服务——Header 对、Range 支持稳、路径看起来像真文件、服务器不截断连接。少一个环节,iOS 或 Chrome 就可能静音、卡住、报 ERR_CONTENT_LENGTH_MISMATCH。











