FastAPI Vue文件分片上传秒传断点续传,如何实现?
摘要:还在为大文件上传失败而抓狂吗?本文不讲天书,只分享一名全栈程序媛在实战中趟过的坑。从原理到代码,手把手带你用FastAPI和Vue实现分片上传、秒传与断点续传。读完你会发现,搞定大文件上传,原来只需要换个思路。
你是不是也遇到过,上传个几百M的安装包或者设计稿,眼瞅着进度条走到99%,然后……网络波动一下,啪,没了!又要从头再来?相信我,这种崩溃的感觉我太懂了。
最近刚好在弄文件上传的小工具,又把这块硬骨头啃了一遍。今天不聊虚的,就把我这次用 FastAPI 和 Vue 实现大文件分片上传的完整思路,还有踩过的坑,都倒出来给你听。
🎯 核心摘要:读完你能带走什么?
这篇文章不会给你扔一堆晦涩的概念就跑。我会用讲故事的方式,跟你讲明白:
👉 分片上传到底是怎么一回事儿,原理是啥。
👉 秒传和断点续传的核心逻辑(其实就隔一层窗户纸)。
👉 一份可以直接复制粘贴拿去用的 FastAPI 后端核心代码。
👉 一份 Vue3 前端的实现思路和关键代码片段。
🗺️ 主要内容脉络
1️⃣ 从一次抓狂的上传失败说起
2️⃣ 别再盯着进度条了:聊聊分片、哈希与秒传
3️⃣ 后端实战:FastAPI如何像收快递一样收文件
4️⃣ 前端实战:Vue里那把切文件的“菜刀”
5️⃣ 跑起来!以及那些年我踩过的坑
6️⃣ 结尾碎碎念
1️⃣ 从一次抓狂的上传失败说起
手机上传一段视频到电脑,“每次到一半就挂” 我一看日志, 413 Request Entity Too Large ,Nginx和FastAPI双双拦截。
调整配置大小限制?那是治标不治本,大文件上传慢、易失败是HTTP协议天生的短板。
这就是为啥我们需要 分片上传。
思路特简单:把大象放进冰箱要几步?不对,是把大文件切成小块,一块块传,最后在服务端拼起来。
2️⃣ 别再盯着进度条了:聊聊分片、哈希与秒传
好,咱们先来理顺几个核心概念,别急着写代码。
🔹 文件切片(Blob.slice)
浏览器里,我们可以用 file.slice() 方法,像切西瓜一样把一个 File 对象切成一个个 Blob 。比如设定每片5MB,一个100MB的文件就变成了20个小块。
🔹 文件哈希(Spark-MD5)
接下来重点来了!文件指纹,也就是MD5值。 这玩意儿是文件的唯一身份证号。你可能会问,为啥要算这个?
一是为了 秒传:后端一看,“哟,这个MD5我数据库里有啊,文件存着呢!” 直接告诉你“传完了”,用户体验瞬间起飞。
二是为了 校验:确保服务端拼起来的文件没缺胳膊少腿。
这里千万别学我当初偷懒,想着用 文件名+修改时间 来当唯一标识,结果用户把文件重命名一下,或者换个文件夹上传,后端就傻傻地又存了一份一模一样的垃圾数据。
🔹 断点续传
原理就是每次上传前,先拿MD5去问后端:“我这文件,哪些片你已经有了?”
后端返回一个数组,比如 [0, 1, 3] ,意思就是第0、1、3片传过了。前端直接跳过这些片,从第2片开始传。
是不是以为这样就完了?对,核心就这么简单!
3️⃣ 后端实战:FastAPI如何像收快递一样收文件
咱们后端用的是 Python 界的当红炸子鸡 FastAPI,它处理文件上传异步非阻塞,性能杠杠的。
📍 检查分片接口(实现秒传+断点续传)
@app.post("/upload/check")
async def check_chunks(file_hash: str, total_chunks: int):
# 1. 去数据库查这个hash
file_record = await db.get_file_by_hash(file_hash)
if file_record:
# 如果存在完整文件记录,直接返回秒传信号
return {"status": "success", "code": 200, "data": {"uploaded": True}}
# 2. 不存在,就去磁盘找临时分片文件夹
temp_dir = Path(f"temp/{file_hash}")
uploaded_chunks = []
if temp_dir.exists():
for chunk_file in temp_dir.iterdir():
# 文件名约定为 chunk_index
uploaded_chunks.append(int(chunk_file.stem))
return {"status": "success", "data": {"uploaded": False, "uploaded_chunks": uploaded_chunks}}
再说个容易翻车的点: 临时文件夹的路径设计。
