.php文件改名为.mp4后无法播放,因为服务器仍执行PHP脚本并返回非MP4内容,而播放器依赖Content-Type和二进制签名(如ftypmp4)识别视频;正确做法是PHP设置video/mp4头并用readfile输出真实MP4流,或通过X-Sendfile/X-Accel-Redirect交由Web服务器处理。

PHP 后缀的文件本身不是视频,它只是服务器端脚本;直接把 .php 文件改成 .mp4 后缀,浏览器和播放器根本不会去执行 PHP,也不会识别为合法 MP4 文件——所以必然播放不了。
为什么改后缀后浏览器打不开或报错?
浏览器请求一个资源时,只看响应头里的 Content-Type 和实际二进制内容。你把 video.php 改成 video.mp4,但服务器仍按 PHP 脚本执行,返回的是 PHP 输出(可能是 HTML、错误信息、甚至空内容),而不是标准 MP4 的二进制流(以 ftypmp4 开头等)。播放器收到非 MP4 数据,直接拒绝解析。
- 常见现象:Chrome 显示 “Failed to load resource: net::ERR_CONTENT_DECODING_FAILED” 或直接空白
- 用
curl -I your-video.mp4查看响应头,大概率是Content-Type: text/html或application/x-httpd-php,而非video/mp4 - 用十六进制工具(如
xxd)查看文件开头,不是00 00 00 18 66 74 79 70(MP4 签名),而是 PHP 源码或 HTML 文本
想用 PHP 动态输出 MP4,必须这么做
PHP 要“伪装”成视频,核心是:不输出任何额外字符,正确设置响应头,并逐块读取真实 MP4 文件(或生成流)输出。漏掉任一环节都会失败。
- 必须用
readfile()或fopen()+fpassthru(),避免内存加载整个大文件 - 必须设置:
header('Content-Type: video/mp4');、header('Accept-Ranges: bytes');、header('Content-Length: ' . filesize($real_mp4_path)); - 禁止在 PHP 文件顶部/底部有空格、BOM、
echo、var_dump等任何输出(否则 HTTP 头发送失败) - 如果要支持拖拽(seek),还得处理
Range请求(需解析$_SERVER['HTTP_RANGE']并返回206 Partial Content)
header('Content-Type: video/mp4');
header('Accept-Ranges: bytes');
header('Content-Length: ' . filesize('/path/to/real-video.mp4'));
readfile('/path/to/real-video.mp4');
更安全的做法:用 Web 服务器直接服务 MP4,PHP 只做权限控制
硬让 PHP 输出大视频文件,容易超时、占内存、不支持断点续传。生产环境推荐把 MP4 放在 Web 根目录外(如 /var/www/private/videos/),由 PHP 验证权限后,用 header('X-Sendfile: /path/to/file.mp4')(Apache + mod_xsendfile)或 X-Accel-Redirect(Nginx)交由 Web 服务器传输——既安全又高效。
立即学习“PHP免费学习笔记(深入)”;
- Nginx 示例配置:
location /protected/ { internal; alias /var/www/private/; } - PHP 中只需:
header('X-Accel-Redirect: /protected/video.mp4');,之后立刻exit; - 这样 PHP 不碰文件内容,Web 服务器负责 Range、缓存、压缩等所有细节
真正卡住人的,往往不是“怎么写 PHP”,而是没意识到:MP4 播放依赖的是字节流格式 + HTTP 协议支持(尤其是 Range),不是文件后缀。后缀只是个提示,服务器响应才是关键。











