PHP无法直接控制视频预加载,其作用仅限于正确输出视频文件、生成带preload属性的HTML或代理流时保留关键Header;必须支持Range请求并返回206状态码、Content-Range和Accept-Ranges头,否则拖拽和预加载失效。
PHP 本身不直接控制视频预加载,因为预加载是浏览器端行为,由 HTML 标签的 preload 属性和 HTTP 服务响应共同决定。PHP 的作用仅限于:正确输出视频文件(尤其是支持范围请求)、生成带合适属性的 HTML、或通过后端代理透传视频流时保留关键 Header。
readfile() 播放视频会卡顿或无法拖拽常见错误是用 PHP 脚本读取视频文件并直接输出,例如:
header('Content-Type: video/mp4');
readfile('/path/to/video.mp4');这会导致两个致命问题:
Accept-Ranges: bytes 和 Content-Range 响应头,浏览器无法发起分片请求 → 拖动进度条失败、无法预加载后续片段Range 请求头,浏览器发来的字节范围请求被忽略 → 视频只能从头播,不能“边下边播”Range 请求才能启用真正预加载要让浏览器能预加载、拖拽、缓冲,PHP 输出必须模拟静态文件服务器行为。核心是解析 $_SERVER['HTTP_RANGE'],计算起始/结束偏移,并返回 206 Partial Content 状态码和对应字节流。
关键点:
Range 请求头,否则 preload="auto" 形同虚设Accept-Ranges: bytes 响应头(即使无 Range 请求也建议返回)fopen(..., 'rb') + fseek() + fread() 流式读取,避免 file_get_contents() 内存爆炸Range 值是否合法(如负数、超出文件大小),防止 500 错误或信息泄露以下代码可部署为 video.php?src=xxx.mp4,支持完整 Range 请求流程:
if (!isset($_GET['src'])) {
http_response_code(400);
exit;
}
$filepath = '/var/www/videos/' . basename($_GET['src']);
if (!is_file($filepath) || !is_readable($filepath)) {
http_response_code(404);
exit;
}
$size = filesize($filepath);
$fp = fopen($filepath, 'rb');
$range = $_SERVER['HTTP_RANGE'] ?? '';
if (preg_match('/^bytes=(\d+)-(\d+)?/', $range, $matches)) {
$start = (int)$matches[1];
$end = isset($matches[2]) ? (int)$matches[2] : $size - 1;
if ($end >= $size) $end = $size - 1;
if ($start > $end || $start < 0) {
http_response_code(416);
header("Content-Range: bytes */$size");
exit;
}
http_response_code(206);
header("Content-Range: bytes $start-$end/$size");
header("Accept-Ranges: bytes");
fseek($fp, $start);
$length = $end - $start + 1;} else {
http_response_code(200);
header("Accept-Ranges: bytes");
$length = $size;
}
header("Content-Type: video/" . pathinfo($filepath, PATHINFO_EXTENSION));
header("Content-Length: $length");
header("Connection: close");
$buffer = 8192;
while (!feof($fp) && $length > 0) {
$read = min($buffer, $length);
echo fread($fp, $read);
$length -= $read;
}
fclose($fp);
使用时在 HTML 中写

:更推荐的做法:别用 PHP 代理,改用 Nginx/Apache 直接托管
PHP 处理大文件流既慢又易出错。生产环境应让 Web 服务器直接服务视频文件:
Range 请求,无需额外配置(确保没禁用 disable_symlinks 或加了不当 rewrite)mod_headers 和 mod_mime,并确认 AcceptPathInfo Off(防绕过)auth_request 模块 + PHP 鉴权接口,而非用 PHP 读文件Accept-Ranges 和 Cache-Control,CDN 才能缓存分片真正容易被忽略的是:哪怕你写了完美的 PHP Range 处理,只要前端 标签没加 preload="auto" 或浏览器隐私模式限制了自动预加载,效果就归零。预加载不是后端单方面能决定的事。