溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

利用斷點續(xù)傳實現(xiàn)下載的原理是什么

發(fā)布時間:2020-12-04 15:36:29 來源:億速云 閱讀:245 作者:Leah 欄目:編程語言

今天就跟大家聊聊有關(guān)利用斷點續(xù)傳實現(xiàn)下載的原理是什么,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。

HTTP斷點續(xù)傳報文

要實現(xiàn)HTTP斷點續(xù)傳必須要簡單了解以下幾個報文。

Accept-Ranges 告訴客戶端(瀏覽器..)服務(wù)器端支持?jǐn)帱c續(xù)傳 服務(wù)器端返回

Range 客戶端告訴服務(wù)器端從指定的的位置/范圍(這里值字節(jié)數(shù))下載資源 客戶端發(fā)出

Content-Range 服務(wù)器端告訴客戶端響應(yīng)的數(shù)據(jù)信息,在整個返回體中本部分的字節(jié)位置 服務(wù)器端返回

ETag 資源標(biāo)識 非必須 服務(wù)器端返回

Last-Modified 資源最后一次更新的時間 非必須 服務(wù)器端返回

Range 的范圍格式

表示0-499個字節(jié)范圍:Range: bytes=0-499

表示最后500個字節(jié)范圍:Range: bytes=-500

表示500字節(jié)開始到結(jié)束范圍:Range: bytes=500-

表示第一個和最后一個字節(jié):Range: bytes=0-0,-1

表示同時指定幾個范圍:Range: bytes=500-600,601-999

Content-Range 的數(shù)據(jù)格式

Content-Range: bytes 0-499/22036 :表示返回0-499字節(jié)范圍數(shù)據(jù) 資源一共22036個字節(jié)

原理

客戶端發(fā)起請求 設(shè)置Range指定開始字節(jié)數(shù)或結(jié)束字節(jié)數(shù) 如果是從0開始也可以不用設(shè)置。

服務(wù)器端檢查到客戶端Range頭 解析開始字節(jié)數(shù)以及結(jié)束字節(jié)數(shù) 并返回報文頭 Accept-Ranges表示支持?jǐn)帱c續(xù)傳,Content-Range記錄該次向客戶端寫入流的位置信息,然后再寫入流到客戶端。

服務(wù)端可以使用ETag Last-Modified 標(biāo)記一下資源是否被修改。作一些驗證工作,如果驗證不通過則返回錯誤,非必須項。

java實現(xiàn)

OutputStream os=null;
 InputStream inputStream =null;
 File zipFile=null;
 try{
  long zipStart=System.currentTimeMillis();
  zipFile=createFile();//動態(tài)根據(jù)業(yè)務(wù)創(chuàng)建文件
  if(logger.isInfoEnabled()){
   logger.info(String.format("壓縮ZIP 花費時間 %s(s) ",
  (System.currentTimeMillis()-zipStart)/1000));
  }
  if (zipFile.exists()) {
   long downloadStart=System.currentTimeMillis();
   inputStream= new BufferedInputStream(new FileInputStream(zipFile));
   response.reset();
   os=new BufferedOutputStream(response.getOutputStream());
   String userAgent = request.getHeader("USER-AGENT");
   String fileName=zipFile.getName();
   if (null != userAgent && -1 != userAgent.indexOf("MSIE")) {
    fileName = URLEncoder.encode(fileName, "UTF8");
   } else if (null != userAgent && -1 != userAgent.indexOf("Mozilla")) {
    fileName = new String(fileName.getBytes("utf-8"), "ISO-8859-1");
   }
   response.setHeader("Accept-Ranges", "bytes");
   response.setHeader("Content-Disposition", 
  "attachment;filename="+ fileName);
   response.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE);
   long pos = 0, fileSize=zipFile.length(),
 last=fileSize-1;
   response.setHeader("ETag",zipFile.getName().
   concat(Objects.toString(fileSize))
     .concat("_").concat(Objects.toString(zipFile.lastModified())));
   response.setDateHeader("Last-Modified",zipFile.lastModified());
   response.setDateHeader("Expires",
   System.currentTimeMillis()+1000*60*60*24);
   if (null != request.getHeader("Range")) {
    response.setStatus(HttpServletResponse.SC_PARTIAL_CONTENT);
    try {
     // 暫時只處理這2種range格式 1、RANGE: bytes=111- 2、Range: bytes=0-499
     String numRang = request.getHeader("Range")
   .replaceAll("bytes=", "");
     String[] strRange = numRang.split("-");
     if (strRange.length == 2) {
      pos = Long.parseLong(strRange[0].trim());
      last = Long.parseLong(strRange[1].trim());
     } else {
      pos = Long.parseLong(numRang.replaceAll("-", "").trim());
     }
    } catch (NumberFormatException e) {
     logger.error(request.getHeader("Range") + " error");
     pos = 0;
    }
   }
   long rangLength = last - pos + 1;
   String contentRange = new StringBuffer("bytes ").
   append(String.valueOf(pos)).
   append("-").append(last).append("/").
   append(String.valueOf(fileSize)).toString();
   response.setHeader("Content-Range", contentRange);
   response.addHeader("Content-Length",Objects.toString(rangLength));
   if(pos>0){
    inputStream.skip(pos);
   }
   byte[] buffer = new byte[1024*512];//每次以512KB 0.5MB的流量下載
   int length = 0,sendTotal=0;
   while (sendTotal < rangLength && length!=-1) {
    length = inputStream.read(buffer, 0,
  ((rangLength - sendTotal) <= buffer.length &#63;
      ((int) (rangLength - sendTotal)) : buffer.length));
    sendTotal = sendTotal + length;
    os.write(buffer, 0, length);
   }
   if(os!=null){
    os.flush();
   }
   if(logger.isInfoEnabled()){
    logger.info(String.format("下載 花費時間 %s(s) ",
  (System.currentTimeMillis()-downloadStart)/1000));
   }
  }
 }catch (Exception e){
  if(StringUtils.endsWithIgnoreCase(e.getMessage(),"Broken pipe")){
   logger.error("用戶取消下載");
  }
  logger.error(e.getMessage(),e);
 }finally {
  if(os!=null){
   try{
    os.close();
   }catch (Exception e){}
  }
  if(inputStream!=null){
   try{
    IOUtils.closeQuietly(inputStream);
   }catch (Exception e){}
  }
 }
}

比如google瀏覽器下載的時候就能看到下載進度以及暫停下載和恢復(fù)下載操作,也可以設(shè)置Range測試分段下載。

看完上述內(nèi)容,你們對利用斷點續(xù)傳實現(xiàn)下載的原理是什么有進一步的了解嗎?如果還想了解更多知識或者相關(guān)內(nèi)容,請關(guān)注億速云行業(yè)資訊頻道,感謝大家的支持。

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI