背景:  DAS为用户提供快速导入数据的功能,允许用户上传最大为15M的SQL/CSV文件

一、原始阶段

一提到文件上传,首先想到的是使用最原始的html的input标签,把类型设置为file就可以了。

<!-- 核心代码 -->
<form action="upload" method="POST" enctype="multipart/form-data"><input type="file" /><input type="submit" />
</form>

优点:原生HTML,简单粗暴,没有任何炫技

缺点:样子过于陈旧,与DAS的中后台系统的设计格格不入

1.1  第1次进化(美化)

HTML的label标签for属性:表示lable标签要绑定的HTML元素,点击这个标签的时候,所绑定的元素将获取焦点。(以前这个特点经常用来在点击checkbox后面的文字时,选中/反选checkbox),所以对于文件选择器一样适用,利用这一点可以实现第1次进化(样式的进化)

.inputfile {width: 0.1px;height: 0.1px;opacity: 0;overflow: hidden;position: absolute;z-index: -1;
}
.inputfile + label {font-size: 14px;font-weight: 400;color: white;background-color: #23c6c8;...
}
<input type="file" id="uploader" accept=".sql" class="inputfile" />
<label for="uploader">上传...<label>

为了把藏起来,把它的宽高都设置成0.1px,透明度设置为0等等无所不用其极,再精心的修改label标签的样式,使其符合DAS的风格,这样文件上传就完成了第1次进化。

优点:仍然是HTML原生支持,只是使用了一点小技巧,可以通过定义css使其适配目标系统的风格

缺点:对于一些较大的文件,上传过程会卡死(或者是接口超时都还没有上传完)

1.2   第2次进化(分段上传)

导入的文件大小限制在15M时,线上表现一直很好。直到有很多用户提出15M的限制太小了,于是DAS开始支持最大1G文件的导入。

实现思路:

  • 调用接口获取1个UploadID

  • 读入文件,然后把文件切割成10M/个的分段(最大103个分段)

  • 上传所有的分段所有分段都完成后,通知服务端根据uploadID合并文件

这个思路的核心在于如何在前端进行文件的分割:

  • HTML5提供的  FileReader  API提供了读取用户本地文件的能力,FileReader 对象允许Web应用程序异步读取存储在用户计算机上的文件(或原始数据缓冲区)的内容,使用File 或  Blob 对象指定要读取的文件或数据。

  • Blob 对象表示一个不可变、原始数据的类文件对象。Blob 表示的不一定是JavaScript原生格式的数据。File 接口基于Blob,继承了 blob 的功能并将其扩展使其支持用户系统上的文件。

  • Blob.slice(start, end, contentTyp)

    返回一个新的 Blob 对象,包含了源 Blob 对象中指定范围内的数据。

核心伪代码:

let chunks = Math.ceil(size / CHUNK_SIZE);
let fileReader = new FileReader();fileReader.onload = e => {let fileData = new Blob([e.target.result]);        currentChunk++;if (currentChunk < chunks) {/* 文件分片读完了,马上开始调接口上传文件 */uploadToServer(fileData).then(()=>{/* 读取下一个分片 */loadNext();}).catch(err=>{/* 错误处理 */})       }
};function loadNext() {var start = currentChunk * CHUNK_SIZE,end = start + CHUNK_SIZE >= file.size ? file.size : start + CHUNK_SIZE;fileReader.readAsArrayBuffer(blobSlice.call(file, start, end));
}loadNext();

在实现了分段上传后,1G文件不会再产阻塞的问题,可以顺利的上传完成,上传的过程如下图所示:

1.3  第3次进化(多轮分段读取 + 前端并发控制)

既然已经支持1G了,为了拉开和竟品的差距(DMS支持最大100M),DAS索性直接支持10G文件的导入。简单进行分段上传的问题开始显现出来。

  • 当只支持1G的时候,把1G文件切分后的。10G的文件全部读入是不可能的(用户内存可能都没有10G)

  • 10G文件最大有1000+个分片,顺序上传的效率又太低,必须采用并发上传的方式

对于第1个问题还比较容易修改,只需要加入最大入的分段数的控制即可,这样就变成了多轮分段读取的方式了,核心代码如下:

fileReader.onload = e => {let fileData = new Blob([e.target.result]);/* 把文件分片保存在组件中 */fileParts.push(fileData);currentChunk++;if (currentChunk < chunks && fileParts.length < 12) {  // 加入一个判断的条件/* 文件分片还没有读完,继续读 */loadNext();} else {        fileParts.forEach(parts=>{/* 文件分片读完了,开始调接口上传文件 */})// 分段上传完成后,把存储分段的数组清空即可。fileParts = [];// 重新开始新一轮的读取loadNext()}
};

第2个问题,如果要采用并发上传的方式,就必须引入一个并发控机制,确保同时上传分段数最多不超过3个,实现并发控制其实也比较简单,受数组的map方法启发,极简版本的实现核心代码如下:

    let recursion = (arr) => {return asyncHandle(arr.shift()).then(()=>{if (arr.length!==0) return recursion(arr) // 数组未迭代完,递归继续else return 'finish';})};let listCopy = [].concat(list);let asyncList = []; // 正在进行的所有并发异步操作while(limit--) {asyncList.push( recursion(listCopy) ); }return Promise.all(asyncList);  // 所有并发异步操作都完成后,本次并发控制迭代完成
}// 测试
var dataLists = [1,2,3,4,5,6,7,8,9,11,100,123,321,789,987];
var count = 0;
mapLimit(dataLists, 3, (curItem)=>{return new Promise(resolve => {count++setTimeout(()=>{console.log(curItem, '当前并发请求:', count--)resolve();}, Math.random() * 5000)  });
}).then(response => {console.log('finish', response)
})

两者结合起来的伪代码:

fileReader.onload = e => {let fileData = new Blob([e.target.result]);/* 把文件分片保存在组件中 */fileParts.push(fileData);currentChunk++;if (currentChunk < chunks && fileParts.length < 12) {  // 加入一个判断的条件/* 文件分片还没有读完,继续读 */loadNext();} else {        mapLimit(fileParts, 3, async (file,callback)=>{await uploadToServer(file).then(()=>{// 分段上传完成后,把存储分段的数组清空即可。fileParts = [];// 重新开始新一轮的读取loadNext();}).catch(err=>{// 预留给下一次进化})                     })        }
};

1.4 第4次进化(重试 )

虽然流式读取和前端并发控制已经很大程度上改善了过程,但是然后有不足之处:

  • 一旦1000+中任何一个分片出错了,整个上传就失败了。这一点没有好好利用分段数据的优势:哪个分段上传不成功,重新上传该分段即可,不用重新上传整个文件。

于是可以在上传过程中发生错误的时候进行重试,当然重试也不应该是无限制的(DAS设计为重试5次,如果某个分片上传失败后,重试5次仍然失败,整个上传过程就认为失败了),核心代码如下:

fileReader.onload = e => {let fileData = new Blob([e.target.result]);/* 把文件分片保存在组件中 */fileParts.push(fileData);currentChunk++;if (currentChunk < chunks && fileParts.length < 12) {  // 加入一个判断的条件/* 文件分片还没有读完,继续读 */loadNext();} else {        mapLimit(fileParts, 3, async (file,callback)=>{await uploadToServer(file).then(()=>{// 分段上传完成后,把存储分段的数组清空即可。fileParts = [];// 重新开始新一轮的读取loadNext();}).catch(async err=>{// 进行5次重试let retryCounts = 0;for (let i = 0; i < MAX_RETRY; i++) {retryCounts++;let [retryErr, retryData] = await uploadToServer(file).then(rs => {                       return [null, rs.data];}.catch(retryErr => [retryErr]);if (retryErr) {/* 重试仍然挂了,继续重试 */continue;} else {/* 重试成功了 */callback(null, retryData);break;}}})                     })        }
};

1.5 第5次进度(支持中途取消)

即使进行了几次优化,但是上传10G文件仍然需要花好久,所以需要允许用户取消文件上传;中途取消也分为两个部分:

  • 分片读取中断,不再发送新的分片上传请求

  • 正在上传中的分片取消请求

第1个诉求只需要在代码中加入标记位cancelFlag即可,用户点取消时,将cancelFlag设置为true即可,核心代码如下:

   function loadNext() {if(!this.state.cancelFlag) {var start = currentChunk * CHUNK_SIZE,end = start + CHUNK_SIZE >= file.size ? file.size : start + CHUNK_SIZE;fileReader.readAsArrayBuffer(blobSlice.call(file, start, end));}}

第2个诉求目前还在实装中,主要原理是基于Cancelable Promises Proposal 实现请求的取消。核心代码如下:

 const CancelToken = axios.CancelToken;const source = CancelToken.source();post('/upload', {cancelToken: source.token    //请求体或者头部传递一个cancelToken}).catch(function (thrown) {if (axios.isCancel(thrown)) {console.log('Request canceled', thrown.message);} else {// handle error}});source.cancel('Operation canceled by the user.');

二、重构的效果

15M同步上传 -> 1G分段顺序上传 -> 10G多轮分段并发上传

  • 最大支持10G文件上传

  • 上传效率高

  • 自动重试

  • 可中途取消上传

【华为云技术分享】DAS文件上传组件的进化相关推荐

  1. 华为云对象存储obs文件上传

    搭建华为云obs服务 开通obs服务 创建桶对象 注意:不用特地买资源包,学习使用的话按需付费即可[记得在自己的华为云账号充一点钱方便他自己扣费] 创建成功之后: 点击进去之后,就可以知道对我们的有用 ...

  2. 基于华为云obs的springMVC文件上传下载,简单demo

    使用华为云的obs作为文件服务 使用springdata jpa框架操作数据库 创建springboot项目,添加华为云obs的SDK的maven依赖 <!-- 开启spring配置类 --&g ...

  3. 【华为云技术分享】三大前端技术(React,Vue,Angular)探密(下)

    [华为云技术分享]三大前端技术(React,Vue,Angular)探密(上) [Angular] Angular(通常被称为 "Angular 2+"或 "Angula ...

  4. 【华为云技术分享】“技术-经济范式”视角下的开源软件演进剖析-part 1

    前言 以互联网为代表的信息技术的迅猛发展对整个经济体系产生了巨大的影响.信息技术的发展一方面使知识的积累和传播更加迅速,知识爆炸性的增长:另一方面,使信息的获取变得越来越容易,信息交流的强度逐渐增加, ...

  5. 【华为云技术分享】“技术-经济范式”视角下的开源软件演进剖析-part 3

    4. 微观层面 4.1 个体动机 在开源软件发展之初, 商业组织的投入很少甚至没有, 完全是靠Richard Stallman 或者 linus Torvalds 这样的个人在努力推动开源软件艰难前行 ...

  6. 阿里云OSS直传多文件上传遇到的问题及解决方案

    本人萌新,刚实习不久,在上一个项目需求中,需要用到阿里云的文件直传服务,通过各种找,最终找了一个比较靠谱的demo,基于plupload插件的一个前端文件上传插件.然后自己再进行了二次封装,并对其中的 ...

  7. php验证码大全(实例分享),php文件上传代码大全(实例分...-php验证码大全(实例分享)-php打印倒三角的实例代码_169IT.COM...

    本节主要内容: php中的文件上传代码 在我们平时的php编程中,涉及文件上传的内容很多,无论是简单的留言本程序,还是复杂的新闻系统,甚至是功能完备的cms系统中,都少不了文件上传的功能与代码. 本文 ...

  8. 微信头像下载并上传到阿里云OSS,PHP文件上传到阿里云OSS简单代码(OSS文件上传,微信头像下载,CURL下载文件,微信头像链接过期)

    (就这么个小事,有多少公司多少项目没做到!!) 微信公众号项目,后端获取到授权用户的微信头像后,要自行下载保存,不下载的话,微信返回的头像链接会在一段时间后过期,无法访问! 下面是我写的两个简单实用方 ...

  9. python程序发布到阿里云云服务器_Python实现阿里云服务器里的文件上传与下载

    Python实现阿里云服务器里的文件上传与下载 018.4.15 背景: 老实说,因为现实的各种原因造成电脑换来换去是可能出现的事情,但是电脑能换,电脑里的环境却不能换.我就曾在三个电脑里各自安装了虚 ...

最新文章

  1. WordPress插件制作教程概述
  2. SAP歷史更改記錄函數
  3. 详解图示+例题演练——BF算法+KMP算法基本原理
  4. electron sqlite3_electron集成sqlite3,win10上折腾了2天
  5. oracle 没有索引删除一行数据_Oracle数据库之索引
  6. javascript绘制静态或者动态的图表、关系表、流程图-JointJS
  7. 高等代数——大学高等代数课程创新教材(丘维声)——3.6笔记+习题
  8. 趣谈网络协议(一):综述及二层到三层
  9. 如何恢复删除的文件?wps文件恢复,4种方法教你找回来
  10. Masscan工具使用
  11. HTTP 405 错误 – 方法不被允许 (Method not allowed)
  12. 因为高考是最相对公平的一次竞争和选拔
  13. 我喜欢的一篇关于家庭教育的文章
  14. 3.3 初学者不能回避——《逆袭大学》连载
  15. Vagrant的安装和使用(附带安装Centos 7教程)
  16. 【jzoj4668】【腐败】【数论】【快速乘】
  17. WM8960驱动的移植记录
  18. 光影精灵5装双系统遇到的问题
  19. Python面向对象二(第十二讲)
  20. 基于粒子群优化算法的无人机路径规划与轨迹算法的实现(Matlab代码实现)

热门文章

  1. ci mysql 缓存_CodeIgniter框架中启用和清除缓存的教程
  2. 视觉SLAM笔记(18) Sophus
  3. 在ROS-melodic中安装map_server、gmapping 等功能包
  4. php curl无视ssl,用Curl实现Post和Get请求,可绕过SSL验证
  5. java for android的书_JavaForAndroid07
  6. mysql temporary_mysql – 如何在同一个查询中多次引用TEMPORARY表?
  7. codeblocks printf函数打印不出来_最全C语言基本程序交互函数之输出到屏幕
  8. springboot 后台模板_spring boot实战
  9. string常用函数用法集合
  10. Nginx http 视频点播服务器搭建操作指南