如何实现大文件的切片上传与断点续传

背景

在我们实际开发中,难免会遇到需要使用上传文件的功能。比较小的文件我们也许不会考虑文件的切片上传,一旦涉及文件比较大,例如几十、上百MB,甚至是GB级别的文件,这种情况下又将如何呢?传输时间将会很长,一旦中途传输产生错误,之前传输的内容都将失效,从头再来。

功能设计

首先思考是两个主体功能,即切片上传与断点续传。在上传过程中肯定会占用用户大量的带宽,再考虑用户再上传中可能会临时处理其他问题,需要暂停上传,于是我们思考合计需要增加暂停与继续上传功能。此场景下,上传需要一定量的时间,所以还要增加上传进度的功能。

如何实现文件切片呢?首先,我们可以通过 input[type=file] 获取用户勾选的文件,它是包含一系列 File 对象的 FileList 对象。这里我们取 files[0] 即可。在 MDN 的介绍中可以了解到,File 对象是特殊类型的 Blob,且可以用在任意的 Blob 类型的 context 中。比如说, FileReader, URL.createObjectURL(), createImageBitmap() (en-US), 及 XMLHttpRequest.send() 都能处理 BlobFile。这里我们可以按照一定大小,将选择的文件切成一系列的 chunks ,然后再以此上传,待上传完毕后,由服务端将 chunks 合并完整文件。

如何实现断点续传呢?由于服务端无法感知到我们上传的是什么文件,将文件名发给服务端验证也不可靠,因为你无法确定服务端是否有同名文件。在此基础上,我们可以先计算出文件的 MD5 值交给服务端去查询是否有相同的文件,文件是否传输完毕,已经上传了多少 chunks 。这时可能有小伙伴提出来 MD5 已经发现有碰撞,用它验证文件的唯一性有一定概率出错。这时我们可以考虑同时搭配其他方法共同验证,比如 SHA1 ,使用 MD5 + SHA1 已经足够确保文件的唯一性。这里我们主要讲思路,后面的例子就统一使用 MD5 来举例。

那又如何实现暂停与继续上传呢?这个就比较简单了。上传 chunks 的时候,我们通常使用递归或循环来做。由此想到可以设置一个标识位,当标识为 true 时就暂停后续上传,当点击继续时,重置标识位,同时重新调用上传逻辑。

功能实现

首先我们需要安装 JavaScript-MD5 到项目中。

shell 复制代码
npm install blueimp-md5

页面上需要一个文件选择框,一个功能按钮。

html 复制代码
<input type="file" name="file" id="file">
<button id="upload" onClick="upload()">upload</button>

上传逻辑。

js 复制代码
import md5 from 'blueimp-md5';

let md5hash = '';
let flag = false;
const chunkSize = 1 * 1024 * 1024;

const input = document.getElementById('file');
const file = input.files[0];

const fr = new FileReader();

fr.addEventListener(
    "load",
    () => {
      md5hash = md5(fr.result);
    },
    false,
);

fr.readAsText(file);

const verifyFile = () => {
    return new Promise((resolve, reject) => {
        const xhr = new XMLHttpRequest();
        xhr.open('POST', 'https://example.com/files/verify', true);
        xhr.onload = (response) => {
            resolve(response);
        };
        xhr.send(md5hash);
    });
};

const uploadChunk = (fileChunk, chunkName) => {
    return new Promise((resolve, reject) => {
        const formData = new FormData()
        formData.append('file', fileChunk, chunkName);

        const xhr = new XMLHttpRequest();
        xhr.open('POST', 'https://example.com/files/upload', true);
        xhr.onload = (response) => {
            resolve(response);
        };
        xhr.send(formData);
    });
};

const upload = async () => {
    const response = await verifyFile();
    const { data }  = response ?? {};
    const { chunkIndex, done } = data ?? {};
    const totalChunks = parseInt(file.size / chunkSize);
    let index = chunkIndex || 0;

    while (!done && !flag && index < totalChunks) {
        count chunk = file.slice(chunkSize * index, chunkSize);
        const chunkName = file.name + 'chunk_' + index;
        
        await uploadChunk(chunk, chunkName);
    };
};

参考文献或文章

相关推荐
特严赤傲20 小时前
H5 页面在微信浏览器里调用微信支付 demo
javascript·微信·jsapi
qq_4019673820 小时前
element-plus table组件 封装列隐藏功能,并非 v-if 或 v-for,通过tableRef 与样式控制
javascript·vue.js·elementui
徐同保20 小时前
react组件内添加一个全局点击时间,点击函数能区分是否是某个特定的id的dom触发的
前端·javascript·react.js
前端 贾公子1 天前
v-if 与 v-for 的优先级对比
开发语言·前端·javascript
小二·1 天前
Pinia 完全指南:用 TypeScript 构建可维护、可测试、可持久化的 Vue 3 状态管理
javascript·vue.js·typescript
bug总结1 天前
Vue3 实现后台管理系统跳转大屏自动登录功能
前端·javascript·vue.js
小二·1 天前
Vue 3 组件通信全方案详解:Props/Emit、provide/inject、事件总线替代与组合式函数封装
前端·javascript·vue.js
Moment1 天前
如何在前端编辑器中实现像 Ctrl + Z 一样的撤销和重做
前端·javascript·面试
小猪猪屁1 天前
权限封装不是写个指令那么简单:一次真实项目的反思
前端·javascript·vue.js
我的写法有点潮1 天前
如何取消Vue Watch监听
前端·javascript·vue.js