JavaScript大文件上傳的處理方法之切片上傳
前言
本篇介紹了切片上傳的基本實現(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實現(xiàn)菜單的打開和關(guān)閉效果demo
本文通過實例代碼給大家分享在不使用JavaScript實現(xiàn)菜單的打開和關(guān)閉效果,非常不錯,具有參考借鑒價值,需要的朋友參考下吧2018-05-05微信小程序?qū)崿F(xiàn)的canvas合成圖片功能示例
這篇文章主要介紹了微信小程序?qū)崿F(xiàn)的canvas合成圖片功能,結(jié)合實例形式分析了微信小程序canvas合成圖片相關(guān)組件使用、操作步驟與注意事項,需要的朋友可以參考下2019-05-05Next.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 程序,對于提高 JavaScript 程序的性能這個問題,最簡單同時也是很容易被忽視的方法就是學(xué)習(xí)如何正確編寫高性能循環(huán)語句。下面我們來學(xué)習(xí)一下吧2019-06-06