欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

JavaScript大文件上傳的處理方法之切片上傳

 更新時間:2022年06月28日 10:28:15   作者:? Dddusty?  ?  
這篇文章主要介紹了JavaScript大文件上傳的處理方法之切片上傳,切片上傳的原理較為簡單,即獲取文件后切片,切片后整理好每個切片的參數(shù)并發(fā)請求即可

前言

本篇介紹了切片上傳的基本實現(xiàn)方式(前端),以及實現(xiàn)切片上傳后的一些附加功能,切片上傳原理較為簡單,代碼注釋比較清晰就不多贅述了,后面的附加功能介紹了實現(xiàn)原理,并貼出了在原本代碼上的改進方式。有什么錯誤希望大佬可以指出,感激不盡。

切片后上傳

切片上傳的原理較為簡單,即獲取文件后切片,切片后整理好每個切片的參數(shù)并發(fā)請求即可。

下面直接上代碼:

HTML

<template>
  <div>
    <input type="file" @change="handleFileChange" />
    <el-button @click="handleUpload">上傳</el-button>
  </div>
</template>

JavaScript

<script>
const SIZE = 10 * 1024 * 1024; // 切片大小

export default {
  data: () => ({
    // 存放文件信息
    container: {
      file: null
      hash: null
    },
    data: [] // 用于存放加工好的文件切片列表
    hashPercentage: 0 // 存放hash生成進度
  }),
  methods: {
    // 獲取上傳文件
    handleFileChange(e) {
      const [file] = e.target.files;
      if (!file) {
        this.container.file = null;
        return;
      }
      this.container.file = file;
    },
        
    // 生成文件切片
    createFileChunk(file, size = SIZE) {
     const fileChunkList = [];
      let cur = 0;
      while (cur < file.size) {
        fileChunkList.push({ file: file.slice(cur, cur + size) });
        cur += size;
      }
      return fileChunkList;
    },
        
    // 生成文件hash    
    calculateHash(fileChunkList) {
      return new Promise(resolve => {
        this.container.worker = new Worker("/hash.js");
        this.container.worker.postMessage({ fileChunkList });
        this.container.worker.onmessage = e => {
          const { percentage, hash } = e.data;
          // 可以用來顯示進度條
          this.hashPercentage = percentage;
          if (hash) {
            resolve(hash);
          }
        };
      });
    },

   	// 切片加工(上傳前預(yù)處理 為文件添加hash等)
    async handleUpload() {
      if (!this.container.file) return;
      // 切片生成
      const fileChunkList = this.createFileChunk(this.container.file);
      // hash生成
      this.container.hash = await this.calculateHash(fileChunkList);
      this.data = fileChunkList.map(({ file },index) => ({
        chunk: file,
        // 這里的hash為文件名 + 切片序號,也可以用md5對文件進行加密獲取唯一hash值來代替文件名
        hash: this.container.hash + "-" + index
      }));
      await this.uploadChunks();
    }
      
    // 上傳切片
   	async uploadChunks() {
     const requestList = this.data
     	// 構(gòu)造formData
       .map(({ chunk,hash }) => {
         const formData = new FormData();
         formData.append("chunk", chunk);
         formData.append("hash", hash);
         formData.append("filename", this.container.file.name);
         return { formData };
       })
     	// 發(fā)送請求 上傳切片
       .map(async ({ formData }) =>
       	request(formData)
       );
     await Promise.all(requestList); // 等待全部切片上傳完畢
     await merge(this.container.file.name) // 發(fā)送請求合并文件
   	},
  }
};
</script>

生成hash

無論是前端還是服務(wù)端,都必須要生成文件和切片的 hash,之前我們使用文件名 + 切片下標作為切片 hash,這樣做文件名一旦修改就失去了效果,而事實上只要文件內(nèi)容不變,hash 就不應(yīng)該變化,所以正確的做法是根據(jù)文件內(nèi)容生成 hash,所以我們修改一下 hash 的生成規(guī)則

這里用到另一個庫 spark-md5,它可以根據(jù)文件內(nèi)容計算出文件的 hash 值,另外考慮到如果上傳一個超大文件,讀取文件內(nèi)容計算 hash 是非常耗費時間的,并且會引起 UI 的阻塞,導(dǎo)致頁面假死狀態(tài),所以我們使用 web-worker 在 worker 線程計算 hash,這樣用戶仍可以在主界面正常的交互

由于實例化 web-worker 時,參數(shù)是一個 js 文件路徑且不能跨域,所以我們單獨創(chuàng)建一個 hash.js 文件放在 public 目錄下,另外在 worker 中也是不允許訪問 dom 的,但它提供了importScripts`函數(shù)用于導(dǎo)入外部腳本,通過它導(dǎo)入 spark-md5

// /public/hash.js
self.importScripts("/spark-md5.min.js"); // 導(dǎo)入腳本
// 生成文件 hash
self.onmessage = e => {
  const { fileChunkList } = e.data;
  const spark = new self.SparkMD5.ArrayBuffer();
  let percentage = 0;
  let count = 0;
  const loadNext = index => {
    // 新建讀取器
    const reader = new FileReader();
    // 設(shè)定讀取數(shù)據(jù)格式并開始讀取
    reader.readAsArrayBuffer(fileChunkList[index].file);
    // 監(jiān)聽讀取完成
    reader.onload = e => {
      count++;
      // 獲取讀取結(jié)果并交給spark計算hash
      spark.append(e.target.result);
      if (count === fileChunkList.length) {
        self.postMessage({
          percentage: 100,
          // 獲取最終hash
          hash: spark.end()
        });
        self.close();
      } else {
        percentage += 100 / fileChunkList.length;
        self.postMessage({
          percentage
        });
        // 遞歸計算下一個切片
        loadNext(count);
      }
    };
  };
  loadNext(0);
};

小結(jié)

  • 獲取上傳文件
  • 文件切片后存入數(shù)組 fileChunkList.push({ file: file.slice(cur, cur + size) });
  • 生成文件hash(非必須)
  • 根據(jù)文件切片列表生成請求列表
  • 并發(fā)請求
  • 待全部請求完成后發(fā)送合并請求

文件秒傳

實際是障眼法,用來欺騙用戶的。

原理:在文件上傳之前先計算出文件的hash,然后發(fā)送給后端進行驗證,看后端是否存在這個hash,如果存在,則證明這個文件上傳過,則直接提示用戶秒傳成功

// 切片加工(上傳前預(yù)處理 為文件添加hash等)
async handleUpload() {
  if (!this.container.file) return;
  // 切片生成
  const fileChunkList = this.createFileChunk(this.container.file);
  // hash生成
  this.container.hash = await this.calculateHash(fileChunkList);
    
  // hash驗證 (verify為后端驗證接口請求)
  const { haveExisetd } = await verify(this.container.hash)
  // 判斷
  if(haveExisetd) {
  	this.$message.success("秒傳:上傳成功") 
    return   
  } 
   
  this.data = fileChunkList.map(({ file },index) => ({
    chunk: file,
    // 這里的hash為文件名 + 切片序號,也可以用md5對文件進行加密獲取唯一hash值來代替文件名
    hash: this.container.hash + "-" + index
  }));
  await this.uploadChunks();
}

暫停上傳

原理:將所有的切片存在一個數(shù)組中,每當一個切片上傳完畢,從數(shù)組中移除,這樣就可以實現(xiàn)用一個數(shù)組只保存上傳中的文件。此外,因為要暫停上傳,所以需要中斷請求 axios中斷請求可以利用AbortController

中斷請求示例

const controller = new AbortController()
axios({
  signal: controller.signal
}).then(() => {});
// 取消請求
controller.abort()

添加暫停上傳功能

// 上傳切片
async uploadChunks() {
 // 需要把requestList放到全局,因為要通過操控requestList來實現(xiàn)中斷
 this.requestList = this.data
 	// 構(gòu)造formData
   .map(({ chunk,hash }) => {
     const formData = new FormData();
     formData.append("chunk", chunk);
     formData.append("hash", hash);
     formData.append("filename", this.container.file.name);
     return { formData };
   })
 	// 發(fā)送請求 上傳切片
   .map(async ({ formData }, index) =>
     request(formData).then(() => {
       // 將請求成功的請求剝離出requestList
       this.requestList.splice(index, 1)
     })
   );
 await Promise.all(this.requestList); // 等待全部切片上傳完畢
 await merge(this.container.file.name) // 發(fā)送請求合并文件
},
// 暫停上傳   
handlePause() {
	this.requestList.forEach((req) => {
        // 為每個請求新建一個AbortController實例
     	const controller = new AbortController();
        req.signal = controller.signal
        controller.abort()
    })
}

恢復(fù)上傳

原理:上傳切片之前,向后臺發(fā)送請求,接口將已上傳的切片列表返回,通過切片hash將后臺已存在的切片過濾,只上傳未存在的切片

// 切片加工(上傳前預(yù)處理 為文件添加hash等)
async handleUpload() {
  if (!this.container.file) return;
  // 切片生成
  const fileChunkList = this.createFileChunk(this.container.file);
  // 文件hash生成
  this.container.hash = await this.calculateHash(fileChunkList);
  // hash驗證 (verify為后端驗證接口請求)
  const { haveExisetd, uploadedList } = await verify(this.container.hash)
  // 判斷
  if(haveExisetd) {
  	this.$message.success("秒傳:上傳成功") 
    return   
  } 
  this.data = fileChunkList.map(({ file },index) => ({
    chunk: file,
    // 注:這個是切片hash   這里的hash為文件名 + 切片序號,也可以用md5對文件進行加密獲取唯一hash值來代替文件名
    hash: this.container.hash + "-" + index
  }));
  await this.uploadChunks(uploadedList);
}
// 上傳切片
async uploadChunks(uploadedList = []) {
 // 需要把requestList放到全局,因為要通過操控requestList來實現(xiàn)中斷
 this.requestList = this.data
    // 過濾出來未上傳的切片
   .filter(({ hash }) => !uploadedList.includes(hash))
 	// 構(gòu)造formData
   .map(({ chunk,hash }) => {
     const formData = new FormData();
     formData.append("chunk", chunk);
     formData.append("hash", hash);
     formData.append("filename", this.container.file.name);
     return { formData };
   })
 	// 發(fā)送請求 上傳切片
   .map(async ({ formData }, index) =>
     request(formData).then(() => {
       // 將請求成功的請求剝離出requestList
       this.requestList.splice(index, 1)
     })
   );
 await Promise.all(this.requestList); // 等待全部切片上傳完畢
 // 合并之前添加一層驗證 驗證全部切片傳送完畢
 if(uploadedList.length + this.requestList.length == this.data.length){
	await merge(this.container.file.name) // 發(fā)送請求合并文件
 }
},
// 暫停上傳   
handlePause() {
	this.requestList.forEach((req) => {
        // 為每個請求新建一個AbortController實例
     	const controller = new AbortController();
        req.signal = controller.signal
        controller.abort()
    })
}
// 恢復(fù)上傳
async handleRecovery() {
    //獲取已上傳切片列表 (verify為后端驗證接口請求)
	const { uploadedList } = await verify(this.container.hash)
    await uploadChunks(uploadedList)
}

添加功能總結(jié)

  • 1.文件秒傳其實就是一個簡單的驗證,把文件的hash發(fā)送給后端,后端驗證是否存在該文件后將結(jié)果返回,如果存在則提示文件秒傳成功
  • 2.斷點傳送分為兩步,暫停上傳和恢復(fù)上傳。暫停上傳是通過獲取到未上傳完畢切片列表(完整切片列表剝離請求已完成的切片后形成),對列表請求進行請求中斷實現(xiàn)的?;謴?fù)上傳實質(zhì)也是一層驗證,在上傳文件之前,將文件的hash發(fā)送給后端,后端返回已經(jīng)上傳完畢的切片列表,然后根據(jù)切片hash將后端返回的切片列表中的切片過濾出去,只上傳未上傳完成的切片。

到此這篇關(guān)于JavaScript大文件上傳的處理方法之切片上傳的文章就介紹到這了,更多相關(guān)JS切片上傳內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • javascript中sort()的用法實例分析

    javascript中sort()的用法實例分析

    這篇文章主要介紹了javascript中sort()的用法,實例分析了sort()的功能、定義及使用技巧,需要的朋友可以參考下
    2015-01-01
  • JS如何輸出26個英文字符

    JS如何輸出26個英文字符

    這篇文章主要介紹了JS如何輸出26個英文字符問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-10-10
  • 簡單談?wù)刯avascript Date類型

    簡單談?wù)刯avascript Date類型

    Date對象,是操作日期和時間的對象。Date對象對日期和時間的操作只能通過方法。本文就給大家簡單講述一下DATE類型的使用。
    2015-09-09
  • 不使用JavaScript實現(xiàn)菜單的打開和關(guān)閉效果demo

    不使用JavaScript實現(xiàn)菜單的打開和關(guān)閉效果demo

    本文通過實例代碼給大家分享在不使用JavaScript實現(xiàn)菜單的打開和關(guān)閉效果,非常不錯,具有參考借鑒價值,需要的朋友參考下吧
    2018-05-05
  • 微信小程序?qū)崿F(xiàn)的canvas合成圖片功能示例

    微信小程序?qū)崿F(xiàn)的canvas合成圖片功能示例

    這篇文章主要介紹了微信小程序?qū)崿F(xiàn)的canvas合成圖片功能,結(jié)合實例形式分析了微信小程序canvas合成圖片相關(guān)組件使用、操作步驟與注意事項,需要的朋友可以參考下
    2019-05-05
  • JavaScript大文件上傳的處理方法之切片上傳

    JavaScript大文件上傳的處理方法之切片上傳

    這篇文章主要介紹了JavaScript大文件上傳的處理方法之切片上傳,切片上傳的原理較為簡單,即獲取文件后切片,切片后整理好每個切片的參數(shù)并發(fā)請求即可
    2022-06-06
  • JS中symbol的特點和用法詳解

    JS中symbol的特點和用法詳解

    在 JavaScript 中,Symbol用于表示獨一無二的標識符,它是 ECMAScript 6引入的一個新特性,本文介紹了JS中的symbol的特點和用法,需要的朋友可以參考下
    2023-06-06
  • Next.js應(yīng)用轉(zhuǎn)換為TypeScript方法demo

    Next.js應(yīng)用轉(zhuǎn)換為TypeScript方法demo

    這篇文章主要為大家介紹了Next.js應(yīng)用轉(zhuǎn)換為TypeScript方法demo,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-12-12
  • 通過循環(huán)優(yōu)化 JavaScript 程序

    通過循環(huán)優(yōu)化 JavaScript 程序

    這篇文章主要介紹了通過循環(huán)優(yōu)化 JavaScript 程序,對于提高 JavaScript 程序的性能這個問題,最簡單同時也是很容易被忽視的方法就是學(xué)習(xí)如何正確編寫高性能循環(huán)語句。下面我們來學(xué)習(xí)一下吧
    2019-06-06
  • uniapp添加車牌組件的實現(xiàn)與使用

    uniapp添加車牌組件的實現(xiàn)與使用

    uniapp是2019年火爆的一個Dcloud開發(fā)跨平臺前端工具,下面這篇文章主要給大家介紹了關(guān)于uniapp添加車牌組件的實現(xiàn)與使用,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-05-05

最新評論