php模拟post请求断点续传_phppost大文件续传法【步骤】

3次阅读

php curl断点续传需客户端分块上传+服务端配合:切片文件、记录偏移、HEAD查已传长度、fseek定位读取、持久化状态;服务端须支持Content-Range、追加写入、返回206/200及正确响应头,并调大PHP和nginx上传限制。

php模拟post请求断点续传_phppost大文件续传法【步骤】

PHP cURL POST 大文件时如何支持断点续传

PHP 原生不支持 http 断点续传(Range 协议)的自动恢复,必须手动控制分块上传 + 服务端配合。关键不是“模拟 POST”,而是把大文件切片、记录已传偏移、失败后从 Content-Range 指定位置继续发。

cURL 设置分块上传和 Range 头的必要参数

服务端必须支持 Content-Range 和返回 206 Partial Content 或自定义成功标识;PHP 端需禁用自动重定向、关闭 body 自动编码,并显式设置头:

  • curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'POST') —— 避免 GET 覆盖
  • curl_setopt($ch, CURLOPT_POSTFIELDS, $chunk_data) —— 只传当前块,非整个文件
  • curl_setopt($ch, CURLOPT_HTTPHEADER, ['Content-Range: bytes '.$start.'-'.$end.'/'.$total_size'])
  • curl_setopt($ch, CURLOPT_HEADER, true) —— 方便读取响应头判断是否接受该块
  • curl_setopt($ch, CURLOPT_RETURNTRANSFER, true) —— 必须开启,否则无法解析响应

如何检测上传中断并恢复(客户端逻辑)

不能依赖 cURL 的 CURLOPT_CONNECTTIMEOUTCURLOPT_TIMEOUT 直接重试整文件——那样等于重头来。要靠状态持久化 + 块级校验:

  • 每次上传前先发 HEAD 请求,查服务端已存长度(如接口返回 X-Uploaded-Bytes: 10485760
  • fopen($file, 'rb') + fseek($fp, $offset) 定位读取,避免内存爆掉
  • 上传失败后,把当前 $start 写入临时文件(如 /tmp/upload_state_abc123.json),含 filenameoffsettotal_size
  • 恢复时读该文件,跳过已传部分,从 $offset 开始下一块

服务端必须做的三件事(否则 PHP 客户端再努力也白搭)

断点续传是双向协议,服务端没配合,客户端所有 Range 头都无效:

立即学习PHP免费学习笔记(深入)”;

  • 接收时检查 HTTP_CONTENT_RANGE,提取 bytes 1000-1999/10000000 中的起止位置
  • FILE_appEND | LOCK_EX 方式写入目标文件,且写之前 fseek($fp, $start)
  • 返回明确状态:上传完成时返回 200 OK + {"status":"success"};中间块返回 206200 + {"received":2000}

最容易被忽略的是:PHP 默认 post_max_sizeupload_max_filesize 会拦截大 POST 体,即使你只传 1MB 块,也要确保这些值 ≥ 单块大小;Nginx 还要调 client_max_body_size

text=ZqhQzanResources