PHP无法直接用fopen读取远程视频,因allow_url_fopen默认关闭且不支持Range请求;必须用cURL透传HTTP_RANGE头、返回206状态码及Content-Range响应头,实现流式分片传输。
fopen() 读取视频?默认情况下,fopen('https://example.com/video.mp4', 'r') 会失败——不是语法错,而是 PHP 的 allow_url_fopen 默认关闭,且即使开启,也仅

cURL 发起带 Range 头的流式请求浏览器播放器(如 HTML5 )首次加载时会发 Range: bytes=0-,拖动进度条则发类似 Range: bytes=1048576- 的请求。PHP 后端必须能响应这类请求,否则无法拖拽、卡顿、报错 ERR_CONTENT_LENGTH_MISMATCH。
CURLOPT_HTTPHEADER 传递客户端原始 Range 头CURLOPT_HEADER,但需手动解析并返回正确的 Content-Range、Accept-Ranges 和状态码(206 Partial Content)CURLOPT_WRITEFUNCTION 流式写入响应体,避免缓存整段视频Range:用 curl -I https://xxx.mp4 看响应头是否有 Accept-Ranges: bytes
header('Content-Type: video/mp4');
header('Accept-Ranges: bytes');
$ch = curl_init($_GET['url']);
curl_setopt($ch, CURLOPT_HEADER, false);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, false);
curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true);
curl_setopt($ch, CURLOPT_TIMEOUT, 30);
if (isset($_SERVER['HTTP_RANGE'])) {
curl_setopt($ch, CURLOPT_HTTPHEADER, [$_SERVER['HTTP_RANGE']]);
header('HTTP/1.1 206 Partial Content');
}
curl_setopt($ch, CURLOPT_WRITEFUNCTION, function($ch, $data) {
echo $data;
return strlen($data);
});
curl_exec($ch);
curl_close($ch);
file_get_contents() 和 readfile() 为什么不适合远程视频?这两个函数底层都依赖 allow_url_fopen,且会把整个远程文件加载进内存(或临时文件),对百 MB 视频就是灾难。尤其 readfile() 虽边读边输出,但无法透传 Range,也无法控制 HTTP 状态码和响应头,浏览器收不到 206,就认定加载失败。
file_get_contents():默认无超时,大文件直接超时或 OOMreadfile():不支持自定义请求头,无法转发 Range
Content-Length 或 Content-Range,导致前端视频控件显示异常如果视频存在阿里云 OSS 或 AWS S3,其直链天然支持 Range,此时 PHP 只需做透明代理,但必须注意:
Referer 或 User-Agent,需在 cURL 中补全:curl_setopt($ch, CURLOPT_REFERER, $_SERVER['HTTP_REFERER'] ?? '')
Expires + Signature 参数),不能简单拼接 URLproxy_buffering off; + proxy_cache off; + 正确透传 Range 和 If-Range
Range 支持,或忽略远程服务端是否真正响应分片,是绝大多数人卡住的地方。
来电咨询