直接用readfile()输出视频会卡顿,因php阻塞读取全文件致内存高、响应慢、不支持边下边播;应实现http Range分片支持,并优先交由nginx等Web服务器处理静态视频。

为什么直接用 readfile() 输出视频会卡顿
PHP 默认以阻塞方式读取整个视频文件再输出,内存占用高、响应慢,尤其在大文件或并发请求时,服务器容易超时或返回 500。浏览器也无法边下边播,必须等全部传输完才开始解码。
- 避免用
readfile()或file_get_contents()直接输出视频二进制 - 禁用输出缓冲:
ob_end_clean()和ini_set('output_buffering', 'Off')必须提前调用 - 务必设置正确的
Content-Type(如video/mp4)和Accept-Ranges: bytes,否则浏览器无法发起分片请求 - 不支持
Range请求的脚本,会导致 ios safari 和部分安卓播放器拒绝播放
如何用 PHP 正确实现 HTTP Range 支持
浏览器拖动进度条、暂停续播、快进等操作,都依赖 HTTP 的 Range 请求。PHP 脚本必须解析 $_SERVER['HTTP_RANGE'],计算偏移量,并只输出对应字节段。
header('HTTP/1.1 206 Partial Content'); header('Content-Type: video/mp4'); header('Accept-Ranges: bytes'); header('Content-Transfer-Encoding: binary'); $filepath = '/path/to/video.mp4'; $size = filesize($filepath); $fp = fopen($filepath, 'rb');
if (isset($_SERVER['HTTP_RANGE'])) { list(, $range) = explode('=', $_SERVER['HTTP_RANGE'], 2); if (strpos($range, ',') !== false) { header('HTTP/1.1 416 Requested Range Not Satisfiable'); exit; } list($start, $end) = array_map('intval', explode('-', $range)); $length = $end - $start + 1; fseek($fp, $start); header("Content-Range: bytes $start-$end/$size"); header("Content-Length: $length"); } else { $start = 0; $length = $size; header("Content-Length: $size"); }
while (!feof($fp) && $length > 0 && connection_status() == CONNECTION_NORMAL) { $buf = fread($fp, min(8192, $length)); echo $buf; $length -= strlen($buf); flush(); } fclose($fp);
哪些场景不该用 PHP 做视频服务
PHP 是通用脚本语言,不是媒体服务器。以下情况应绕过 PHP,交给 Web 服务器原生处理:
- 静态视频文件直传:Nginx 可通过
add_header Accept-Ranges bytes;和内置range模块直接支持分片,性能远超 PHP - 需要 HLS/dash 自适应流:PHP 不适合实时切片,应使用
ffmpeg预生成.m3u8+.ts,再由 Nginx/apache 托管 - 高并发点播(>100 QPS):PHP-FPM 进程会迅速耗尽,推荐用 caddy、Nginx + X-Accel-Redirect 或专用流媒体服务(如 Wowza、Nimble streamer)
- 防盗链或鉴权逻辑复杂时:可用 PHP 校验权限后,用
X-Sendfile(Apache)或X-Accel-Redirect(Nginx)触发 Web 服务器内部重定向,避免 PHP 读取文件
PHP 层能做的轻量级优化点
如果必须走 PHP(比如动态权限校验、临时 Token 验证),这些小改动能明显改善首帧加载和拖动体验:
立即学习“PHP免费学习笔记(深入)”;
- 开启
zlib.output_compression = Off—— 视频是二进制,压缩无效且增加 CPU 开销 - 关闭所有非必要扩展(如 xdebug、xhprof),它们会让
fread()延迟显著上升 - 用
fopen(..., 'rb')而非file_get_contents(),减少内存峰值 - 对
mp4文件,确保moovbox 在文件开头(可用ffmpeg -i in.mp4 -c copy -movflags +faststart out.mp4修复),否则即使支持 Range,首帧也要等完整下载
实际部署时,moov 位置和 Web 服务器是否真正启用 range 支持,比 PHP 代码本身更容易被忽略。