1. 程式人生 > 程式設計 >php實現斷點續傳大檔案示例程式碼

php實現斷點續傳大檔案示例程式碼

一、斷點續傳原理

所謂斷點續傳,也就是要從檔案已經下載的地方開始繼續下載。在以前版本的 HTTP 協議是不支援斷點的,HTTP/1.1 開始就支援了。一般斷點下載時才用到 Range 和 Content-Range 實體頭。

不使用斷點續傳

get /down.zip http/1.1
accept: image/gif,image/x-xbitmap,image/jpeg,image/pjpeg,application/vnd.ms-
excel,application/msword,application/vnd.ms-powerpoint,*/*
accept-language: zh-cn
accept-encoding: gzip,deflate
user-agent: mozilla/4.0 (compatible; msie 5.01; windows nt 5.0)
connection: keep-alive

伺服器收到請求後,按要求尋找請求的檔案,提取檔案的資訊,然後返回給瀏覽器,返回資訊如下:

HTTP/1.1 200 Ok
content-length=106786028
accept-ranges=bytes
date=mon,30 apr 2001 12:56:11 gmt
etag=w/"02ca57e173c11:95b"
content-type=application/octet-stream
server=microsoft-iis/5.0
last-modified=mon,30 apr 2001 12:56:11 gmt

使用斷點續傳

GET /down.zip HTTP/1.0
User-Agent: NetFox
RANGE: bytes=2000070-
Accept: text/html,image/gif,*; q=.2,*/*; q=.2

多了這麼一行Range: bytes=2000070-

這一行的意思就是告訴伺服器down.zip這個檔案從2000070位元組開始傳,前面的位元組不用傳了。
Range的完整格式是:

Range: bytes=startOffset-targetOffset/sum [表示從startOffset讀取,一直讀取到targetOffset位置,讀取總數為sum直接]
 
Range: bytes=startOffset-targetOffset [位元組總數也可以去掉]

伺服器收到這個請求以後,返回的資訊如下:

HTTP/1.1 206 Partial Content
content-length=106786028
content-range=bytes 2000070-106786027/106786028
date=mon,30 apr 2001 12:55:20 gmt
etag=w/"02ca57e173c11:95b"
content-type=application/octet-stream
server=microsoft-iis/5.0
last-modified=mon,30 apr 2001 12:55:20 gmt

和前面伺服器返回的資訊比較一下,就會發現增加了一行:

Content-Range=bytes 2000070-106786027/106786028

返回的程式碼也改為206了,而不再是200了。

HTTP/1.1 206 Partial Content

知道了以上原理,就可以進行斷點續傳的程式設計了。

二、PHP實現

/** php下載類,支援斷點續傳
 * download: 下載檔案
 * setSpeed: 設定下載速度
 * getRange: 獲取header中Range
 */
 
class FileDownload{
 
 /** 下載
  * @param String $file 要下載的檔案路徑
  * @param String $name 檔名稱,為空則與下載的檔名稱一樣
  * @param boolean $reload 是否開啟斷點續傳
  */
 public function download($file,$name='',$reload=false){
  $fp = @fopen($file,'rb');
  if($fp){
   if($name==''){
    $name = basename($file);
   }
   $header_array = get_headers($file,true);
   //var_dump($header_array);die;
   // 下載本地檔案,獲取檔案大小
   if (!$header_array) {
    $file_size = filesize($file);
   } else {
    $file_size = $header_array['Content-Length'];
   }
   $ranges = $this->getRange($file_size);
   $ua = $_SERVER["HTTP_USER_AGENT"];//判斷是什麼型別瀏覽器
   header('cache-control:public');
   header('content-type:application/octet-stream'); 
 
   $encoded_filename = urlencode($name);
   $encoded_filename = str_replace("+","%20",$encoded_filename);
 
   //解決下載檔名亂碼
   if (preg_match("/MSIE/",$ua) || preg_match("/Trident/",$ua) ){    
    header('Content-Disposition: attachment; filename="' .$encoded_filename . '"');
   } else if (preg_match("/Firefox/",$ua)) {
    header('Content-Disposition: attachment; filename*="utf8\'\'' . $name . '"');
   }else if (preg_match("/Chrome/",$ua)) {
    header('Content-Disposition: attachment; filename="' . $encoded_filename . '"');
   } else {
    header('Content-Disposition: attachment; filename="' . $name . '"');
   }
   //header('Content-Disposition: attachment; filename="' . $name . '"');
 
   if($reload && $ranges!=null){ // 使用續傳
    header('HTTP/1.1 206 Partial Content');
    header('Accept-Ranges:bytes');
 
    // 剩餘長度
    header(sprintf('content-length:%u',$ranges['end']-$ranges['start']));
 
    // range資訊
    header(sprintf('content-range:bytes %s-%s/%s',$ranges['start'],$ranges['end'],$file_size));
    //file_put_contents('test.log',sprintf('content-length:%u',$ranges['end']-$ranges['start']),FILE_APPEND);
    // fp指標跳到斷點位置
    fseek($fp,sprintf('%u',$ranges['start']));
   }else{
    file_put_contents('test.log','2222',FILE_APPEND);
    header('HTTP/1.1 200 OK');
    header('content-length:'.$file_size);
   }
 
   while(!feof($fp)){
    //echo fread($fp,round($this->_speed*1024,0));
    //echo fread($fp,$file_size);
    echo fread($fp,4096);
    ob_flush();
   }
 
   ($fp!=null) && fclose($fp);
  }else{
   return '';
  }
 }
 
 /** 設定下載速度
  * @param int $speed
  */
 public function setSpeed($speed){
  if(is_numeric($speed) && $speed>16 && $speed<4096){
   $this->_speed = $speed;
  }
 }
 
 /** 獲取header range資訊
  * @param int $file_size 檔案大小
  * @return Array
  */
 private function getRange($file_size){
  //file_put_contents('range.log',json_encode($_SERVER),FILE_APPEND);
  if(isset($_SERVER['HTTP_RANGE']) && !empty($_SERVER['HTTP_RANGE'])){
   $range = $_SERVER['HTTP_RANGE'];
   $range = preg_replace('/[\s|,].*/','',$range);
   $range = explode('-',substr($range,6));
   if(count($range)<2){
    $range[1] = $file_size;
   }
   $range = array_combine(array('start','end'),$range);
   if(empty($range['start'])){
    $range['start'] = 0;
   }
   if(empty($range['end'])){
    $range['end'] = $file_size;
   }
   return $range;
  }
  return null;
 }
}
 
$obj = new FileDownload();
$obj->download('http://down.golaravel.com/laravel/laravel-master.zip',true);

以上就是php實現斷點續傳大檔案示例程式碼的詳細內容,更多關於php 斷點續傳大檔案的資料請關注我們其它相關文章!