稀土掘金技术社区 02月21日
大文件上传| React + NestJs |分片、断点续传、秒传 , 你是否知道 ???
index_new5.html
../../../zaker_core/zaker_tpl_static/wap/tpl_guoji1.html

 

本文深入探讨了前端大文件上传的实现方案,针对传统一次性上传可能导致的问题,提出了分片上传的策略。通过将大文件分割成小块,逐个上传,最后在服务器端合并,有效解决了内存占用和网络不稳定的问题。同时,文章还介绍了如何利用哈希计算实现秒传和断点续传功能,提升用户体验。文章还提供了React和NestJS的代码示例,并对前端切片进行了优化,包括使用web worker多线程切片和IndexedDB储存未完成切片。

🗂️**文件分片上传:** 将大文件分割成小块进行上传,解决了内存占用过高和网络传输不稳定的问题,是处理大文件上传的常用策略。

🔑**哈希计算与秒传:** 通过计算文件的哈希值,可以判断文件是否已经上传过,从而实现秒传功能,提升用户体验。

🔄**断点续传:** 记录已上传的文件块,在网络中断后可以从上次上传的位置继续上传,避免重复上传,节省时间和资源。

🛠️**技术栈:** 前端使用React+TS,后端使用NestJs+TS,涉及axios、spark-md5、antd、Multer等技术,保证了代码的健壮性和可维护性。

原创 浪遏 2025-02-17 08:31 重庆

点击关注公众号,“技术干货”及时达!

点击关注公众号,“技术干货” 及时达!

前言

大文件上传不能一次性将整个大文件直接上传,因为这可能会导致

所以通常的做法是把大文件分割成一个个小块,分别上传这些小块,最后在服务器端将这些小块合并成完整的文件。

同时,为了支持秒传和断点续传功能,还需要对文件进行哈希计算,通过哈希值来判断文件是否已经上传过,以及哪些小块已经上传。

「项目地址」: 大文件上传 demo[1]

前端 vue 和 react 均实现 , 后端 Nest 实现 , 代码可以运行 , 点个赞吧~

「前端效果如图:」

「后端效果如图:」 文件上传 , 之后 , 我保存到 uploads 文件夹下 :

大文件上传流程

和倔友一样 , 文字太多 , 很有压力 ,于是乎画了一张图? , 「不知可否点个赞」 ~

「外层数字序号是上图解析 , 内层序号是实现细节 , 可先看外层数字序号」

    前端选择文件后,计算文件哈希值并分割文件(「文件分片」

前端发送请求到后端检查文件上传状态

前端根据后端返回的状态决策

步骤 3 中 第 1,3 点 ,所有文件块上传完成后,前端发送请求通知后端合并文件块。

后端将文件块合并成完整的文件 , 并先前端响应成功信息

通过上述这种方式,前端和后端相互配合,实现了大文件的上传、秒传和断点续传功能。

接下来 , 具体看看代码如何实现 ~

代码实现

「前端使用 React + TS」

    「axios」: 基于 promise 向后端发送 HTTP 请求

    「spark-md5」 : 使用哈希算法计算文件的哈希值

    「antd」: 引入上传文件的组件

    「TypeScript」 : 增强代码健壮性

「后端使用 NestJs + TS」

    「NestJS」:一个用于构建高效、可靠和可扩展的服务器端应用程序的渐进式 Node.js 框架。

    「Multer」:用于处理 multipart/form-data 的中间件,主要用于上传文件。

    「TypeScript」:NestJs 支持 TS

    「Node.js 文件系统 (fs) 模块」:用于与文件系统交互,如读取、写入和删除文件和目录。

初次之外是一些语法检查工具 ESlint

导库

代码总体结构如下 :

UI

首先写 App 函数 , 实现利用 Upload.Dragger 实现上传 UI

添加处理函数 handChange , 若文件有效 , 进入 handFileUpload 函数中 , 进行

哈希计算与分片

在 handFileUpload 中 , 实现文件分片 , MD5 哈希值计算

计算 MD5 哈希值我们封装的函数如下 : 使用 Spark-md5 库

文件上传状态分流

handFileUpload 计算哈希值 和 实现文件分片后 ,

首先检查文件上传情况 , 向后端发一个请求 , 发送 hash 和 文件名 以及文件块的长度

后端通过 Query 从 params 中获取这三个字段

通过检查 uploads 目录下 , 是否有完整文件或文档块来决定返回'uploaded'(秒传) 还是'uploading'(断点续传)

之后根据这些状态 , 选择合适的处理方法

如果是秒传 , 直接告诉用户成功 ; 未上传状态和部分上传状态 , 会进入下面的代码 , 请求后端:http://localhost:3000/upload[2] 接口 , 在后端创建文件夹 , 并且保存分块

分块合并

最后继续执行前端下面的逻辑 , 请求后端将分块合并

后端的实现如下 :

如此大文件上传流程走完 ~

总结

把上面这张图拿下来 , :

面试官 : 请说说大文件上传 ?

你 : 掏出掘金看完这篇文章 , 开始你的表演 ~

优化点

前端切片

切片后 , 如果用户突然关掉浏览器 , 可以将 blob(二进制大对象) , 储存到 IndexedDB , 下次用户近来时候 ,

嗅探一下是否存在未完成的切片

使用 websocket 双向通信 , 实时通知 和请求序列控制

......

下次 , 在来 , 关注我 , 点个赞 , 我酱油更大的动力 , 用心写文章 !

点击关注公众号,“技术干货” 及时达!

阅读原文

跳转微信打开

Fish AI Reader

Fish AI Reader

AI辅助创作,多种专业模板,深度分析,高质量内容生成。从观点提取到深度思考,FishAI为您提供全方位的创作支持。新版本引入自定义参数,让您的创作更加个性化和精准。

FishAI

FishAI

鱼阅,AI 时代的下一个智能信息助手,助你摆脱信息焦虑

联系邮箱 441953276@qq.com

相关标签

大文件上传 前端开发 断点续传 文件分片
相关文章