PHP无法直接用fopen读取远程视频,因allow_url_fopen默认关闭且不支持Range请求;必须用cURL透传HTTP_RANGE头、返回206状态码及Content-Range响应头,实现流式分片传输。

PHP 远程文件不能直接用 fopen() 读取视频?
默认情况下,fopen('https://example.com/video.mp4', 'r') 会失败——不是语法错,而是 PHP 的 allow_url_fopen 默认关闭,且即使开启,也仅支持基础 HTTP GET,不带请求头、无法处理重定向、无超时控制,更不支持 Range 请求(关键!视频预览依赖分片加载)。直接读取远程视频文件到内存再输出,极易触发内存溢出或超时。
用 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():默认无超时,大文件直接超时或 OOM -
readfile():不支持自定义请求头,无法转发Range - 二者均无法获取远程响应的真实
Content-Length或Content-Range,导致前端视频控件显示异常
CDN 或对象存储(如 OSS、S3)的直链已支持 Range,但要注意 Referer 和签名
如果视频存在阿里云 OSS 或 AWS S3,其直链天然支持 Range,此时 PHP 只需做透明代理,但必须注意:
立即学习“PHP免费学习笔记(深入)”;
- 源站可能校验
Referer或User-Agent,需在 cURL 中补全:curl_setopt($ch, CURLOPT_REFERER, $_SERVER['HTTP_REFERER'] ?? '') - 有时需带临时签名(如 OSS 的
Expires+Signature参数),不能简单拼接 URL - 若用 Nginx 做反向代理,比 PHP 更高效:
proxy_buffering off;+proxy_cache off;+ 正确透传Range和If-Range
Range 支持,或忽略远程服务端是否真正响应分片,是绝大多数人卡住的地方。











