直接用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 代码本身更容易被忽略。











