溫馨提示×

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

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

Vue.Js和Java實(shí)現(xiàn)文件分片上傳的方法

發(fā)布時(shí)間:2020-06-23 15:50:59 來(lái)源:億速云 閱讀:857 作者:清晨 欄目:編程語(yǔ)言

這篇文章將為大家詳細(xì)講解有關(guān)Vue.Js和Java實(shí)現(xiàn)文件分片上傳的方法,小編覺(jué)得挺實(shí)用的,因此分享給大家做個(gè)參考,希望大家閱讀完這篇文章后可以有所收獲。


代碼從項(xiàng)目中剝離修改,未經(jīng)測(cè)試,僅提供思路。

前端

upload(file) {
 //從后臺(tái)獲取已經(jīng)上傳的文件分片數(shù)
 getIdx(md5)
  .then(function(res) {
   let retry = 3;
   uploadPart(retry, file, res.data);
  })
  .catch(); 
}

uploadPart(retry, file, idx) {
 //設(shè)置分片大?。▎挝籅yte)
 let bufferLength = 1024 * 1024 * 5;
 //計(jì)算開(kāi)始的切割點(diǎn),idx是上傳成功的分片數(shù),未上傳過(guò)文件則開(kāi)始點(diǎn)為0
 let start = idx * bufferLength;
 //全部上傳完畢或重試次數(shù)用完則退出
 if(start>=file.size || retry<=0) return;
 //計(jì)算分割的位置
 let end = start + bufferLength;
 //如果分割點(diǎn)超出文件大小,回退分割點(diǎn)
 if (end > file.size) {end = fileSize;}
 //切割文件
 var chunk = file.slice(start, end);
 //創(chuàng)建 formData 對(duì)象并添加數(shù)據(jù)
 let formData = new FormData();
 formData.set("file", chunk);
 //如果是第一次上傳,連同文件塊數(shù)量也上傳
 if (start == 0) { 
  //計(jì)算文件切片總數(shù),向上取整
  let chunkNum = Math.ceil(file.size / bufferLength);
  formData.set("total", chunkNum);
 }
 //上傳文件的api,此處使用axios發(fā)送請(qǐng)求
 doUpload(formData)
  //發(fā)送成功,則上傳下一片,遞歸調(diào)用方法
  .then(function() {
   retry = xx;//刷新重試次數(shù)
   uploadPart(retry, file, ++idx);
  })
  //發(fā)送失敗
  .catch(function() {
   retry--;//重試次數(shù)減一
   //重試上傳這一片
   uploadPart(retry, file, idx);
  });
},

文件分片上傳的前端關(guān)鍵代碼只有一句:

//切割文件
var chunk = file.slice(start, end);

通過(guò)slice方法來(lái)切割文件,然后文件上傳的流程視業(yè)務(wù)和具體技術(shù)而定,此處是使用axios發(fā)送請(qǐng)求,用遞歸調(diào)用上傳文件塊。
需要注意的是,Blob.slice(start, end),文件塊包含start指向的字節(jié),而不包含end指向的字節(jié),在使用時(shí)要注意Blob的邊界。

mozilla對(duì)slice的說(shuō)明

后端

/**合并文件的實(shí)際操作*/
public static void doMergeFiles(String outFile, String[] files) {
  //設(shè)置緩存大小
  int BUFSIZE = 1024 * 1024;
  //排序。文件后綴名是文件的順序。
  Arrays.sort(files);
  //輸出流
  FileChannel outChannel = null;
  //標(biāo)記最后的一個(gè)文件
  String lastFlag = files[files.length-1];
  try {
    outChannel = new FileOutputStream(outFile).getChannel();
    //遍歷文件列表
    for(String f : files){
      //最后一塊文件用真實(shí)大小設(shè)置緩存,避免自動(dòng)填充數(shù)據(jù)造成的md5不一致
      if(lastFlag.equals(f)){
        File last = new File(f);
        BUFSIZE = (int) last.length();//獲取文件的大小并設(shè)置成緩存的大小
      }
      FileChannel fc = new FileInputStream(f).getChannel();
      //用ByteBuffer創(chuàng)建緩存
      ByteBuffer bb = ByteBuffer.allocate(BUFSIZE);
      while(fc.read(bb) != -1){//把數(shù)據(jù)讀到緩存
        bb.flip();//重置游標(biāo)
        outChannel.write(bb);//寫入數(shù)據(jù)
        bb.clear();//清空數(shù)據(jù)
      }
      fc.close();//關(guān)閉流
    }
  } catch (IOException ioe) {
    ioe.printStackTrace();
  } finally {
    try {if (outChannel != null) {outChannel.close();}} catch (IOException ignore) {}
  }
}

后端的關(guān)鍵是合并文件,當(dāng)上傳完最后一塊文件就進(jìn)行文件的合并。使用ByteBuffer緩存,使用FileChannel進(jìn)行文件的讀寫完成合并操作。在保存文件時(shí),文件名取一致,文件的后綴名則取文件塊的順序,比如第一塊文件是“xxx.01”,第10塊是“xxx.10”,注意,個(gè)位數(shù)前面要補(bǔ)“0”,這樣可以直接用Array.sort()進(jìn)行排序。

為提高性能,可以適當(dāng)設(shè)置緩存大小,可以邊上傳文件邊合并,不必等到文件都上傳了才合并。

拓展

此處的文件上傳是一次上傳一片,上傳成功才開(kāi)始上傳下一片。如果前端不是使用javascript,能開(kāi)啟使用多線程的話,可以改成同時(shí)上傳多片文件提高上傳速度。已經(jīng)上傳的文件分片用bitmap存儲(chǔ),上傳文件前,從后臺(tái)獲取已上傳的文件分片的bitmap數(shù)據(jù)然后解析,多線程處理未上傳的文件分片。

關(guān)于Vue.Js和Java實(shí)現(xiàn)文件分片上傳的方法就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到。

向AI問(wèn)一下細(xì)節(jié)

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

AI