Vue2与Vue3在实现大文件断点续传上有何区别?

大文件上传方案探索:从WebUploader到自定义分片上传的实践

作为一名前端开发工程师,最近遇到了一个颇具挑战性的需求:需要在Vue项目中实现4GB左右大文件的稳定上传,且要兼容Chrome、Firefox、Edge等主流浏览器,后端使用PHP接收。此前我们采用了百度开源的WebUploader组件,但在实际使用中遇到了几个难以解决的问题:

  1. 分片上传过程中偶尔会出现断点续传失效的情况
  2. 对新版浏览器的兼容性不够理想
  3. 缺乏官方技术支持,社区活跃度下降
  4. 自定义UI的灵活性不足

方案选型思考

经过技术调研,我评估了以下几个主流方案:

  1. Plupload:功能全面但文档不够友好,对Vue集成支持一般
  2. Uppy:现代感强但体积较大,学习曲线较陡
  3. Resumable.js:专注分片上传但UI较为基础
  4. 自定义实现:基于XMLHttpRequest/Fetch API实现核心分片逻辑

最终决定采用自定义分片上传方案,主要基于以下考虑:

  • 完全控制上传流程
  • 可以针对业务需求深度优化
  • 减少第三方依赖,降低维护成本
  • 与Vue生态无缝集成

核心实现思路

1. 前端分片策略

// 文件分片工具函数constchunkFile=(file,chunkSize=5*1024*1024)=>{constchunks=[]letcurrent=0while(current<file.size){chunks.push({file:file.slice(current,current+chunkSize),chunkIndex:chunks.length,totalChunks:Math.ceil(file.size/chunkSize),fileName:file.name,fileSize:file.size,fileType:file.type,fileLastModified:file.lastModified,identifier:generateFileIdentifier(file)// 生成唯一标识用于断点续传})current+=chunkSize}returnchunks}// 生成文件唯一标识(基于文件内容)constgenerateFileIdentifier=(file)=>{returnnewPromise((resolve)=>{constreader=newFileReader()reader.onload=(e)=>{constarr=newUint8Array(e.target.result)consthashArray=Array.from(arr).map(b=>b.toString(16).padStart(2,'0'))resolve(hashArray.join('').substring(0,16))}reader.readAsArrayBuffer(file.slice(0,1024*1024))// 取前1MB计算哈希})}

2. Vue组件实现

export default { data() { return { file: null, chunks: [], uploadStatus: 'idle', // idle, uploading, paused, completed, error progress: 0, error: null, currentChunk: 0, abortController: null } }, methods: { async handleFileChange(e) { this.file = e.target.files[0] if (!this.file) return // 生成文件标识(简化版,实际项目应使用更可靠的算法) const identifier = await this.generateSimpleIdentifier(this.file) // 检查服务器是否有未完成的上传记录 const res = await this.checkUploadStatus(identifier) if (res.exists) { if (confirm('检测到未完成的上传,是否继续?')) { this.currentChunk = res.uploadedChunks } else { // 清除服务器记录(实际项目应实现) } } this.chunks = this.chunkFile(this.file) this.progress = Math.round((this.currentChunk / this.chunks.length) * 100) }, async startUpload() { if (!this.file) return this.uploadStatus = 'uploading' this.error = null this.abortController = new AbortController() try { for (let i = this.currentChunk; i < this.chunks.length; i++) { if (this.uploadStatus !== 'uploading') break // 处理暂停情况 const chunk = this.chunks[i] const formData = new FormData() formData.append('file', chunk.file) formData.append('chunkIndex', chunk.chunkIndex) formData.append('totalChunks', chunk.totalChunks) formData.append('fileName', chunk.fileName) formData.append('fileSize', chunk.fileSize) formData.append('fileType', chunk.fileType) formData.append('identifier', chunk.identifier) await this.uploadChunk(formData) this.currentChunk = i + 1 this.progress = Math.round(((i + 1) / this.chunks.length) * 100) } if (this.uploadStatus === 'uploading') { await this.mergeChunks(this.chunks[0].identifier, this.chunks[0].fileName) this.uploadStatus = 'completed' this.$emit('upload-complete') } } catch (err) { console.error('上传失败:', err) this.error = err.message || '上传过程中出现错误' this.uploadStatus = 'error' } }, async uploadChunk(formData) { return fetch('/api/upload-chunk.php', { method: 'POST', body: formData, signal: this.abortController.signal }) }, async mergeChunks(identifier, fileName) { return fetch('/api/merge-chunks.php', { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ identifier, fileName }) }) }, // 简化版标识生成(实际项目应使用更可靠的算法) generateSimpleIdentifier(file) { return `${file.name}-${file.size}-${file.lastModified}` }, async checkUploadStatus(identifier) { // 实际项目应实现与后端的交互 return { exists: false, uploadedChunks: 0 } }, pauseUpload() { if (this.uploadStatus === 'uploading') { this.abortController.abort() this.uploadStatus = 'paused' } }, resumeUpload() { if (this.uploadStatus === 'paused') { this.startUpload() } } } }

3. PHP后端实现要点

$chunkIndex,'totalChunks'=>$totalChunks,'fileName'=>$fileName,'uploaded'=>time()]));echojson_encode(['status'=>'success']);}else{http_response_code(500);echojson_encode(['status'=>'error','message'=>'Failed to save chunk']);}// merge-chunks.php - 合并分片header('Content-Type: application/json');$uploadDir='/path/to/upload/dir/';$tempDir=$uploadDir.'temp/';$data=json_decode(file_get_contents('php://input'),true);$identifier=$data['identifier']??'';$fileName=$data['fileName']??'';// 检查标识符和文件名if(empty($identifier)||empty($fileName)){http_response_code(400);echojson_encode(['status'=>'error','message'=>'Invalid parameters']);exit;}// 检查上传状态文件$statusFile=$tempDir.$identifier.'.upload';if(!file_exists($statusFile)){http_response_code(404);echojson_encode(['status'=>'error','message'=>'Upload not found']);exit;}$status=json_decode(file_get_contents($statusFile),true);$totalChunks=$status['totalChunks']??0;// 合并文件$finalPath=$uploadDir.$fileName;if($fp=fopen($finalPath,'wb')){for($i=0;$i<$totalChunks;$i++){$chunkPath=$tempDir.$identifier.'_'.$i;if(!file_exists($chunkPath)){fclose($fp);unlink($finalPath);// 删除已创建的部分文件http_response_code(400);echojson_encode(['status'=>'error','message'=>'Missing chunk '.$i]);exit;}$content=file_get_contents($chunkPath);fwrite($fp,$content);unlink($chunkPath);// 删除已合并的分片}fclose($fp);// 删除状态文件unlink($statusFile);echojson_encode(['status'=>'success','path'=>$finalPath]);}else{http_response_code(500);echojson_encode(['status'=>'error','message'=>'Failed to create final file']);}?>

方案优势与改进点

优势

  1. 完全可控:从分片策略到上传逻辑完全自主实现
  2. 深度优化:可以根据网络状况动态调整分片大小
  3. 良好兼容:基于标准Web API实现,兼容所有现代浏览器
  4. 断点续传:通过文件标识实现可靠的断点续传
  5. 进度可视化:精确计算上传进度

可改进方向

  1. 并发上传:当前实现是顺序上传,可优化为并发上传提高速度
  2. 文件校验:增加MD5/SHA校验确保文件完整性
  3. 更可靠的标识生成:当前简化版标识可能存在冲突风险
  4. 服务端清理:实现自动清理未完成上传的临时文件
  5. 拖拽上传:增强用户体验,支持拖放文件上传

实施建议

  1. 渐进式实现:先实现基本分片上传,再逐步添加断点续传、并发上传等功能
  2. 充分测试:在不同网络环境和浏览器下进行全面测试
  3. 监控上报:添加上传失败监控和错误上报机制
  4. 性能优化:根据实际测试结果调整分片大小和并发数

通过这种自定义实现方式,我们成功解决了WebUploader带来的各种问题,同时获得了更好的性能和更灵活的控制能力。目前该方案已在我们项目中稳定运行数月,处理了数百个4GB+文件的上传,未出现重大故障。

将组件复制到项目中

示例中已经包含此目录

引入组件

配置接口地址

接口地址分别对应:文件初始化,文件数据上传,文件进度,文件上传完毕,文件删除,文件夹初始化,文件夹删除,文件列表
参考:http://www.ncmem.com/doc/view.aspx?id=e1f49f3e1d4742e19135e00bd41fa3de

处理事件

启动测试

启动成功

效果

数据库

效果预览

文件上传

文件刷新续传

支持离线保存文件进度,在关闭浏览器,刷新浏览器后进行不丢失,仍然能够继续上传

文件夹上传

支持上传文件夹并保留层级结构,同样支持进度信息离线保存,刷新页面,关闭页面,重启系统不丢失上传进度。

批量下载

支持文件批量下载

下载续传

文件下载支持离线保存进度信息,刷新页面,关闭页面,重启系统均不会丢失进度信息。

文件夹下载

支持下载文件夹,并保留层级结构,不打包,不占用服务器资源。

下载示例

点击下载完整示例

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/1188977.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

6款高效论文辅助软件推荐,附赠专业公式编辑教程

核心工具对比速览 工具名称 核心功能 适用阶段 独特优势 AIbiye 论文结构优化 初稿完成后 理工科逻辑框架自动检测 AIcheck 万字论文生成 开题/初稿 实证研究模块内置 AskPaper 文献综述生成 文献调研 中英文文献混合处理 秒篇 快速论文生成 紧急任务 10分钟…

学术写作利器盘点:6款主流工具+公式编辑资源大全

核心工具对比速览 工具名称 核心功能 适用阶段 独特优势 AIbiye 论文结构优化 初稿完成后 理工科逻辑框架自动检测 AIcheck 万字论文生成 开题/初稿 实证研究模块内置 AskPaper 文献综述生成 文献调研 中英文文献混合处理 秒篇 快速论文生成 紧急任务 10分钟…

6大论文写作平台功能解析,搭配高效公式编辑解决方案

核心工具对比速览 工具名称 核心功能 适用阶段 独特优势 AIbiye 论文结构优化 初稿完成后 理工科逻辑框架自动检测 AIcheck 万字论文生成 开题/初稿 实证研究模块内置 AskPaper 文献综述生成 文献调研 中英文文献混合处理 秒篇 快速论文生成 紧急任务 10分钟…

精选6大论文写作辅助平台,涵盖专业公式编辑工具解析

核心工具对比速览 工具名称 核心功能 适用阶段 独特优势 AIbiye 论文结构优化 初稿完成后 理工科逻辑框架自动检测 AIcheck 万字论文生成 开题/初稿 实证研究模块内置 AskPaper 文献综述生成 文献调研 中英文文献混合处理 秒篇 快速论文生成 紧急任务 10分钟…

红队实战:用 CodeQL + LLM 打造“自动代码审计机”,我在 GitHub 热门项目里挖到了 3 个 0-day

标签&#xff1a; #CodeQL #LLM #RedTeam #0Day #AutomatedAudit #CyberSecurity&#x1fa78; 前言&#xff1a;告别“误报地狱” 每一个做过源码审计的人都知道&#xff0c;使用传统工具扫描时&#xff0c;最痛苦的不是没漏洞&#xff0c;而是99% 的误报。 工具告诉你&#x…

6款热门论文辅助工具详细对比,附带公式编辑实用资源

核心工具对比速览 工具名称 核心功能 适用阶段 独特优势 AIbiye 论文结构优化 初稿完成后 理工科逻辑框架自动检测 AIcheck 万字论文生成 开题/初稿 实证研究模块内置 AskPaper 文献综述生成 文献调研 中英文文献混合处理 秒篇 快速论文生成 紧急任务 10分钟…

智能合约“黑暗森林”:复现 DeFi 重入攻击,AI 竟然比黑客更快发现了合约漏洞?

标签&#xff1a; #Web3 #BlockchainSecurity #Solidity #AI #Reentrancy&#x1fa78; 前言&#xff1a;当 ATM 机发疯的时候 想象一下&#xff0c;你在这个世界上有一台特殊的 ATM 机。 如果你去取 100 块钱&#xff0c;它的流程是这样的&#xff1a; 检查余额&#xff1a;看…

深度解析6款论文辅助工具,提供专业公式编辑技巧指南

核心工具对比速览 工具名称 核心功能 适用阶段 独特优势 AIbiye 论文结构优化 初稿完成后 理工科逻辑框架自动检测 AIcheck 万字论文生成 开题/初稿 实证研究模块内置 AskPaper 文献综述生成 文献调研 中英文文献混合处理 秒篇 快速论文生成 紧急任务 10分钟…

论文写作必备工具清单:6大平台测评与公式编辑资源

核心工具对比速览 工具名称 核心功能 适用阶段 独特优势 AIbiye 论文结构优化 初稿完成后 理工科逻辑框架自动检测 AIcheck 万字论文生成 开题/初稿 实证研究模块内置 AskPaper 文献综述生成 文献调研 中英文文献混合处理 秒篇 快速论文生成 紧急任务 10分钟…

2.网络通信知识点

1、以上仅供参考,如有疑问,留言联系

Rocketmq Dashboard jar 包启动,使用启动命令参数,修改 NameServer 的地址

Rocketmq Dashboard jar 包启动&#xff0c;使用启动命令参数&#xff0c;修改 NameServer 的地址 你可以直接在启动 rocketmq-dashboard 的 JAR 包时&#xff0c;通过命令行参数来指定 NameServer 的地址。这种方式非常灵活&#xff0c;无需修改 JAR 包内部的配置文件。 以下是…

adsldp.dll文件损坏找不到问题 免费下载方法分享

在使用电脑系统时经常会出现丢失找不到某些文件的情况&#xff0c;由于很多常用软件都是采用 Microsoft Visual Studio 编写的&#xff0c;所以这类软件的运行需要依赖微软Visual C运行库&#xff0c;比如像 QQ、迅雷、Adobe 软件等等&#xff0c;如果没有安装VC运行库或者安装…

2026年行业内质量好的截止阀企业哪家好,电动闸阀/暗杆闸阀/铸钢闸阀/手动盲板阀/硬密封球阀,截止阀供应商联系电话 - 品牌推荐师

在工业自动化与能源转型的双重驱动下,不锈钢截止阀作为管道系统中的核心控制元件,其质量直接关系到石油化工、电力能源、冶金制造等重工业领域的安全生产与效率提升。据中国机械工业联合会及《阀门行业白皮书(2025)…

想找口碑好的展柜制作厂电话?文博展示! - 工业品牌热点

本榜单依托全维度市场调研与真实行业口碑,深度筛选出五家标杆企业,为企业选型提供客观依据,助力精准匹配适配的展柜制作伙伴。 TOP1 推荐:成都盛世文博展览展示有限公司 推荐指数:★★★★★ | 口碑评分:西南地区…

AdvancedEmojiDS.dll文件丢失找不到问题 免费下载方法分享

在使用电脑系统时经常会出现丢失找不到某些文件的情况&#xff0c;由于很多常用软件都是采用 Microsoft Visual Studio 编写的&#xff0c;所以这类软件的运行需要依赖微软Visual C运行库&#xff0c;比如像 QQ、迅雷、Adobe 软件等等&#xff0c;如果没有安装VC运行库或者安装…

advapi32.dll文件损坏丢失了 免费下载方法分享

在使用电脑系统时经常会出现丢失找不到某些文件的情况&#xff0c;由于很多常用软件都是采用 Microsoft Visual Studio 编写的&#xff0c;所以这类软件的运行需要依赖微软Visual C运行库&#xff0c;比如像 QQ、迅雷、Adobe 软件等等&#xff0c;如果没有安装VC运行库或者安装…

2026国内最新仿香定制香精生产厂家top5推荐!广东广州优质品牌及厂商全面解析,专业定制服务助力行业创新发展 - 品牌推荐2026

引言 随着消费市场对产品个性化与体验感的需求不断升级,仿香定制香精作为日化、食品、家居等行业的核心配套,其品质、合规性与创新能力成为品牌竞争力的关键。据中国香料香精化妆品工业协会最新行业报告显示,2025年…

advapi32res.dll文件丢失找不到问题 免费下载方法分享

在使用电脑系统时经常会出现丢失找不到某些文件的情况&#xff0c;由于很多常用软件都是采用 Microsoft Visual Studio 编写的&#xff0c;所以这类软件的运行需要依赖微软Visual C运行库&#xff0c;比如像 QQ、迅雷、Adobe 软件等等&#xff0c;如果没有安装VC运行库或者安装…

实用指南:一体化系统(九)智慧社区综合报表——东方仙盟练气期

实用指南:一体化系统(九)智慧社区综合报表——东方仙盟练气期2026-01-20 12:30 tlnshuju 阅读(0) 评论(0) 收藏 举报pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !impor…

深聊西安新华电脑软件学校怎么样,教学特色全知道 - 工业品牌热点

2026年职业教育改革持续深化,选择正规、适配市场需求的技能院校已成为年轻人实现职业跃升的关键支撑。无论是学费透明度、教学质量评估,还是院校正规性甄别,优质职业院校的专业能力直接决定学子的技能掌握程度与未来…