JAVA支持HTTP斷點續傳

jopen 9年前發布 | 2K 次閱讀 Java

文件下載是WEB網站提供的最基本服務,然而你知道HTTP的斷點續傳是怎么實現的嗎?

背景

    這兩天在實現一個基于HTML5在線音視頻播放,由于文件是存放于企業網盤中的,HTTP不可達,因此需要用程序來實現文件的讀取和HTTP協議的下載。

    用Java實現文件下載也不用多說了,讀取文件,通過二進制流的方式往response里寫就行了。H5播放器調用也能進行播放了;然而當我控制進度的前進和后退時,問題來了,居然一點效果都沒有!沒有快進播放器還叫播放器嗎?

分析

    首先看到播放器無法取得音視頻文件的時間長度,很自然想到Content-Length屬性,后臺通過file.length()取得文件長度并設置到Content-Length上(代碼如下),前臺播放器里可以顯示音視頻的長度了,并且可以快進了;然而當我快退的時候,還是無效,同時后臺報錯。

response.addHeader("Content-Length", file.length());



    換了一個HTTP文件進行比較測試,發現直接HTTP訪問的能夠正常快進快退。仔細分析兩者的request和response頭,發現了區別,請求參數 多了如下圖所示屬性,該屬性表名需要從服務端獲取的資源范圍,默認從第一個字節開始取,快進快退實際上就是通過指定這個 Range屬性來確定你所期望的起始點。

    然而這個屬性是在請求頭上的,客戶端又是怎么知道要添加這個屬性呢?繼續尋找發現了 Accept-Ranges這個屬性,屬性值是bytes,其表明是否接受獲取其某個實體的一部分(比如文件的一部分)的請求。bytes:表示接受,none:表示不接受。與此對應的response中另外一個屬性 Content-Range,其表名該響應包含的部分對象為整個對象的哪個部分。完整響應頭如下:


解決

根據上面的分析,我們就知道在服務端該怎么處理了,首先在響應頭上添加Accept-Ranges

response.setHeader("Accept-Ranges", "bytes");

接著判斷request中是否存在Range屬性,即是否指定的起點,若存在,則通過stream的skip直接跳到目標起點,最后添加Content-Range屬性表名當前塊的起始和結束,完整代碼如下:

stream = new FileInputStream(file);
if(request.getHeader("Range") != null) //客戶端請求的下載的文件塊的開始字節   
  {
    //從請求中得到開始的字節   
    //請求的格式是:   
    //Range: bytes=[文件塊的開始字節]-  
    String range = StringUtils.substringBetween(request.getHeader("Range"), "bytes=", "-");
    long start = Long.parseLong(range);   
    //下載的文件(或塊)長度   
    //響應的格式是:   
    //Content-Length: [文件的總大小] - [客戶端請求的下載的文件塊的開始字節]   
    response.setHeader("Content-Length", String.valueOf(fileSize - start));   

    if (start != 0)  
    {
      //要設置狀態   
      //響應的格式是:   
      //HTTP/1.1 206 Partial Content  
               response.setStatus(javax.servlet.http.HttpServletResponse.SC_PARTIAL_CONTENT);//206   
      //不是從最開始下載,   
      //響應的格式是:   
      //Content-Range: bytes [文件塊的開始字節]-[文件的總大小 - 1]/[文件的總大小]   
      response.setHeader("Content-Range","bytes " + start + "-" + String.valueOf(fileSize - 1) + "/" + String.valueOf(fileSize));  

      stream.skip(start);
      }
    }
    responseBinaryStream(response, this.getContentType(FilenameUtils.getExtension(fileName)), stream);


 本文由用戶 jopen 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!