

新闻资讯
技术教程PHP无法异步加载视频,仅能通过返回URL、设置CORS、处理Range请求等方式配合前端video标签实现按需加载;必须支持206响应与faststart优化,否则拖动失败或metadata加载卡顿。
PHP 是服务端脚本语言,它不能直接控制浏览器播放视频,也不具备“异步加载视频资源”的能力。所谓“PHP 异步加载视频”,实际是前端行为——PHP 只能配合生成合适响应(如返回视频 URL、设置 CORS、控制流式传输等)。如果试图用 exec() 或 shell_exec() 在 PHP 中启动 FFmpeg 后再“推流”,那不是“加载播放”,而是自行搭建流媒体服务,复杂度和风险陡增。
真正异步加载的关键配置浏览器对 的加载行为由 HTML 属性和服务器响应共同决定。PHP 要做的,是确保它返回的视频资源支持按需分片请求(即支持 HTTP Range 请求),否则即使写了 preload="none",也可能触发整文件下载。
preload="none":告诉浏览器“先别下视频”,但仅作提示,不强制;preload="metadata" 更常用——只拉头部信息(时长、宽高、编码格式),不加载画面数据Range 请求,返回 206 Partial Content 和 Accept-Ranges: bytes
/video.php?id=123),需手动处理 $_SERVER['HTTP_RANGE'] 并输出对应字节段,否则会返回完整文件+200 OK,导致卡顿和浪费带宽如果出于权限控制、防盗链等目的,必须用 PHP 脚本输出视频(而非直链静态文件),那就得自己实现 Range 支持。漏掉这一步, 拖动、快进、移动端加载都会失败。
header('Content-Type: video/mp4');
header('Accept-Ranges: bytes');
$filepath = '/path/to/video.mp4';
$size = filesize($filepath);
$length = $size;
$start = 0;
$end = $size - 1;
if (isset($_SERVER['HTTP_RANGE'])) {
preg_match('/bytes=(\d+)-(\d+)?/', $_SERVER['HTTP_RANGE'], $matches);
$start = intval($matches[1]);
$end = isset($matches[2]) ? intval($matches[2]) : $size - 1;
$length = $end - $start + 1;
header('HTTP/1.1 206 Partial Content');
header("Content-Range: bytes $start-$end/$size");}
header("Content-Length: $length");
header('Content-Disposition: inline; filename="video.mp4"');
$fp = fopen($filepath, 'rb');
fseek($fp, $start);
while ($length > 0 && !feof($fp)) {
$read = min(8192, $length);
echo fread($fp, $read);
$length -= $read;
flush();
}
fclose($fp);
CORS 和 MIME 类型错误是前端播放失败的高频原因
当视频资源走 PHP 接口(尤其跨域场景),两个看似简单却极易被忽略的点会导致 静音、黑屏或报错 DOMException: The element has no supported sources:
Access-Control-Allow-Origin 响应头 → 浏览器拒绝加载,控制台报 CORS 错误Content-Type(比如 text/html 或空)→ 浏览器不识别为可播放媒体,静默失败moov 原子前置),会导致 preload="metadata" 卡住,因为头部信息在文件末尾用 ffprobe 检查:ffprobe -v quiet -show_entries format=duration -of default=nw=1 input.mp4 若超时或报错,大概率是 moov 位置不对;修复用:ffmpeg -i input.mp4 -c copy -movflags +faststart output.mp4。