原文

以下是本文所涉及到的知识点,break or continue ?

  • 文件上传原理
  • 最原始的文件上传
  • 使用 koa2 作为服务端写一个文件上传接口
  • 单文件上传和上传进度
  • 多文件上传和上传进度
  • 拖拽上传
  • 剪贴板上传
  • 大文件上传之分片上传
  • 大文件上传之断点续传
  • node 端文件上传

原理很简单,就是根据 http 协议的规范和定义,完成请求消息体的封装和消息体的解析,然后将二进制内容保存到文件。

我们都知道如果要上传一个文件,需要把 form 标签的enctype设置为multipart/form-data,同时method必须为post方法。

那么multipart/form-data表示什么呢?

multipart 互联网上的混合资源,就是资源由多种元素组成,form-data 表示可以使用 HTML Forms 和 POST 方法上传文件,具体的定义可以参考 RFC 7578。

multipart/form-data 结构

看下 http 请求的消息体

文件上传下载 - 图1

  • 请求头:

Content-Type: multipart/form-data; boundary=----WebKitFormBoundaryDCntfiXcSkPhS4PN 表示本次请求要上传文件,其中 boundary 表示分隔符,如果要上传多个表单项,就要使用 boundary 分割,每个表单项由———XXX 开始,以———XXX 结尾。

  • 消息体- Form Data 部分

每一个表单项又由Content-TypeContent-Disposition组成。

Content-Disposition: form-data 为固定值,表示一个表单元素,name 表示表单元素的 名称,回车换行后面就是name的值,如果是上传文件就是文件的二进制内容。

Content-Type:表示当前的内容的 MIME 类型,是图片还是文本还是二进制数据。

解析

客户端发送请求到服务器后,服务器会收到请求的消息体,然后对消息体进行解析,解析出哪是普通表单哪些是附件。

可能大家马上能想到通过正则或者字符串处理分割出内容,不过这样是行不通的,二进制buffer转化为string,对字符串进行截取后,其索引和字符串是不一致的,所以结果就不会正确,除非上传的就是字符串。

不过一般情况下不需要自行解析,目前已经有很成熟的三方库可以使用。

至于如何解析,这个也会占用很大篇幅,后面的文章在详细说。

使用 form 表单上传文件

ie时代,如果实现一个无刷新的文件上传那可是费老劲了,大部分都是用 iframe 来实现局部刷新或者使用 flash 插件来搞定,在那个时代 ie 就是最好用的浏览器(别无选择)。

DEMO

文件上传下载 - 图2

这种方式上传文件,不需要 js ,而且没有兼容问题,所有浏览器都支持,就是体验很差,导致页面刷新,页面其他数据丢失。

HTML

  1. <form method="post" action="http://localhost:8100" enctype="multipart/form-data">
  2. 选择文件:
  3. <input type="file" name="f1"/> input 必须设置 name 属性,否则数据无法发送<br/>
  4. <br/>
  5. 标题:<input type="text" name="title"/><br/><br/><br/>
  6. <button type="submit" id="btn-0">上 传</button>
  7. </form>
  8. 复制代码

服务端文件的保存基于现有的库koa-body结合 koa2实现服务端文件的保存和数据的返回。

在项目开发中,文件上传本身和业务无关,代码基本上都可通用。

在这里我们使用koa-body库来实现解析和文件的保存。

koa-body 会自动保存文件到系统临时目录下,也可以指定保存的文件路径。

文件上传下载 - 图3

然后在后续中间件内得到已保存的文件的信息,再做二次处理。

  • ctx.request.files.f1 得到文件信息,f1input file 标签的 name
  • 获得文件的扩展名,重命名文件

NODE

  1. /**
  2. * 服务入口
  3. */
  4. var http = require('http');
  5. var koaStatic = require('koa-static');
  6. var path = require('path');
  7. var koaBody = require('koa-body');//文件保存库
  8. var fs = require('fs');
  9. var Koa = require('koa2');
  10. var app = new Koa();
  11. var port = process.env.PORT || '8100';
  12. var uploadHost= `http://localhost:${port}/uploads/`;
  13. app.use(koaBody({
  14. formidable: {
  15. //设置文件的默认保存目录,不设置则保存在系统临时目录下 os
  16. uploadDir: path.resolve(__dirname, '../static/uploads')
  17. },
  18. multipart: true // 开启文件上传,默认是关闭
  19. }));
  20. //开启静态文件访问
  21. app.use(koaStatic(
  22. path.resolve(__dirname, '../static')
  23. ));
  24. //文件二次处理,修改名称
  25. app.use((ctx) => {
  26. var file = ctx.request.files.f1;//得道文件对象
  27. var path = file.path;
  28. var fname = file.name;//原文件名称
  29. var nextPath = path+fname;
  30. if(file.size>0 && path){
  31. //得到扩展名
  32. var extArr = fname.split('.');
  33. var ext = extArr[extArr.length-1];
  34. var nextPath = path+'.'+ext;
  35. //重命名文件
  36. fs.renameSync(path, nextPath);
  37. }
  38. //以 json 形式输出上传文件地址
  39. ctx.body = `{
  40. "fileUrl":"${uploadHost}${nextPath.slice(nextPath.lastIndexOf('/')+1)}"
  41. }`;
  42. });
  43. /**
  44. * http server
  45. */
  46. var server = http.createServer(app.callback());
  47. server.listen(port);
  48. console.log('demo1 server start ...... ');
  49. 复制代码

CODE

github.com/Bigerfe/fe-…

ie 时代的多文件上传是需要创建多个 input file 标签,现在 html5只需要一个标签加个属性就搞定了,file 标签开启multiple

DEMO

文件上传下载 - 图4

HTML

  1. //设置 multiple属性
  2. <input type="file" name="f1" multiple/>
  3. 复制代码

NODE

服务端也需要进行简单的调整,由单文件对象变为多文件数组,然后进行遍历处理。

  1. //二次处理文件,修改名称
  2. app.use((ctx) => {
  3. var files = ctx.request.files.f1;// 多文件, 得到上传文件的数组
  4. var result=[];
  5. //遍历处理
  6. files && files.forEach(item=>{
  7. var path = item.path;
  8. var fname = item.name;//原文件名称
  9. var nextPath = path + fname;
  10. if (item.size > 0 && path) {
  11. //得到扩展名
  12. var extArr = fname.split('.');
  13. var ext = extArr[extArr.length - 1];
  14. var nextPath = path + '.' + ext;
  15. //重命名文件
  16. fs.renameSync(path, nextPath);
  17. //文件可访问路径放入数组
  18. result.push(uploadHost+ nextPath.slice(nextPath.lastIndexOf('/') + 1));
  19. }
  20. });
  21. //输出 json 结果
  22. ctx.body = `{
  23. "fileUrl":${JSON.stringify(result)}
  24. }`;
  25. })
  26. 复制代码

CODE

github.com/Bigerfe/fe-…

这里说的是在 ie 时代的上传文件局部刷新,借助 iframe 实现。

DEMO

文件上传下载 - 图5

  • 局部刷新

页面内放一个隐藏的 iframe,或者使用 js 动态创建,指定 form 表单的 target 属性值为iframe标签 的 name 属性值,这样 form 表单的 shubmit 行为的跳转就会在 iframe 内完成,整体页面不会刷新。

  • 拿到接口数据

然后为 iframe 添加load事件,得到 iframe 的页面内容,将结果转换为 JSON 对象,这样就拿到了接口的数据

HTML

  1. <iframe id="temp-iframe" name="temp-iframe" src="" style="display:none;"></iframe>
  2. <form method="post" target="temp-iframe" action="http://localhost:8100" enctype="multipart/form-data">
  3. 选择文件(可多选):
  4. <input type="file" name="f1" id="f1" multiple/><br/> input 必须设置 name 属性,否则数据无法发送<br/>
  5. <br/>
  6. 标题:<input type="text" name="title"/><br/><br/><br/>
  7. <button type="submit" id="btn-0">上 传</button>
  8. </form>
  9. <script>
  10. var iframe = document.getElementById('temp-iframe');
  11. iframe.addEventListener('load',function () {
  12. var result = iframe.contentWindow.document.body.innerText;
  13. //接口数据转换为 JSON 对象
  14. var obj = JSON.parse(result);
  15. if(obj && obj.fileUrl.length){
  16. alert('上传成功');
  17. }
  18. console.log(obj);
  19. });
  20. </script>
  21. 复制代码

NODE

服务端代码不需要改动,略.

CODE

github.com/Bigerfe/fe-…

无刷新上传文件肯定要用到XMLHttpRequest,在 ie 时代也有这个对象,单只 支持文本数据的传输,无法用来读取和上传二进制数据。

现在已然升级到了XMLHttpRequest2,较 1 版本有非常大的升级,首先就是可以读取和上传二进制数据,可以使用·FormData·对象管理表单数据。

当然也可使用 fetch 进行上传。

DEMO

文件上传下载 - 图6

HTML

  1. <div>
  2. 选择文件(可多选):
  3. <input type="file" id="f1" multiple/><br/><br/>
  4. <button type="button" id="btn-submit">上 传</button>
  5. </div>
  6. 复制代码

JS xhr

  1. <script>
  2. function submitUpload() {
  3. //获得文件列表,注意这里不是数组,而是对象
  4. var fileList = document.getElementById('f1').files;
  5. if(!fileList.length){
  6. alert('请选择文件');
  7. return;
  8. }
  9. var fd = new FormData(); //构造FormData对象
  10. fd.append('title', document.getElementById('title').value);
  11. //多文件上传需要遍历添加到 fromdata 对象
  12. for(var i =0;i<fileList.length;i++){
  13. fd.append('f1', fileList[i]);//支持多文件上传
  14. }
  15. var xhr = new XMLHttpRequest(); //创建对象
  16. xhr.open('POST', 'http://localhost:8100/', true);
  17. xhr.send(fd);//发送时 Content-Type默认就是: multipart/form-data;
  18. xhr.onreadystatechange = function () {
  19. console.log('state change', xhr.readyState);
  20. if (this.readyState == 4 && this.status == 200) {
  21. var obj = JSON.parse(xhr.responseText); //返回值
  22. console.log(obj);
  23. if(obj.fileUrl.length){
  24. alert('上传成功');
  25. }
  26. }
  27. }
  28. }
  29. //绑定提交事件
  30. document.getElementById('btn-submit').addEventListener('click',submitUpload);
  31. </script>
  32. 复制代码

JS Fetch

  1. fetch('http://localhost:8100/', {
  2. method: 'POST',
  3. body: fd
  4. })
  5. .then(response => response.json())
  6. .then(response =>{
  7. console.log(response);
  8. if (response.fileUrl.length) {
  9. alert('上传成功');
  10. }
  11. } )
  12. .catch(error => console.error('Error:', error));
  13. 复制代码

CODE

github.com/Bigerfe/fe-…

借助XMLHttpRequest2的能力,实现多个文件或者一个文件的上传进度条的显示。

DEMO

文件上传下载 - 图7

说明

  • 页面内增加一个用于显示进度的标签 div.progress
  • js 内处理增加进度处理的监听函数xhr.upload.onprogress
  • event.lengthComputable这是一个状态,表示发送的长度有了变化,可计算
  • event.loaded表示发送了多少字节
  • event.total表示文件总大小
  • 根据event.loadedevent.total计算进度,渲染div.progress

PS 特别提醒

xhr.upload.onprogress要写在xhr.send方法前面,否则event.lengthComputable状态不会改变,只有在最后一次才能获得,也就是100%的时候.

HTML

  1. <div>
  2. 选择文件(可多选):
  3. <input type="file" id="f1" multiple/><br/><br/>
  4. <div id="progress">
  5. <span class="red"></span>
  6. </div>
  7. <button type="button" id="btn-submit">上 传</button>
  8. </div>
  9. 复制代码

JS

  1. <script>
  2. function submitUpload() {
  3. var progressSpan = document.getElementById('progress').firstElementChild;
  4. var fileList = document.getElementById('f1').files;
  5. progressSpan.style.width='0';
  6. progressSpan.classList.remove('green');
  7. if(!fileList.length){
  8. alert('请选择文件');
  9. return;
  10. }
  11. var fd = new FormData(); //构造FormData对象
  12. fd.append('title', document.getElementById('title').value);
  13. for(var i =0;i<fileList.length;i++){
  14. fd.append('f1', fileList[i]);//支持多文件上传
  15. }
  16. var xhr = new XMLHttpRequest(); //创建对象
  17. xhr.open('POST', 'http://10.70.65.235:8100/', true);
  18. xhr.onreadystatechange = function () {
  19. console.log('state change', xhr.readyState);
  20. if (xhr.readyState == 4) {
  21. var obj = JSON.parse(xhr.responseText); //返回值
  22. console.log(obj);
  23. if(obj.fileUrl.length){
  24. //alert('上传成功');
  25. }
  26. }
  27. }
  28. xhr.onprogress=updateProgress;
  29. xhr.upload.onprogress = updateProgress;
  30. function updateProgress(event) {
  31. console.log(event);
  32. if (event.lengthComputable) {
  33. var completedPercent = (event.loaded / event.total * 100).toFixed(2);
  34. progressSpan.style.width= completedPercent+'%';
  35. progressSpan.innerHTML=completedPercent+'%';
  36. if(completedPercent>90){//进度条变色
  37. progressSpan.classList.add('green');
  38. }
  39. console.log('已上传',completedPercent);
  40. }
  41. }
  42. //注意 send 一定要写在最下面,否则 onprogress 只会执行最后一次 也就是100%的时候
  43. xhr.send(fd);//发送时 Content-Type默认就是: multipart/form-data;
  44. }
  45. //绑定提交事件
  46. document.getElementById('btn-submit').addEventListener('click',submitUpload);
  47. </script>
  48. 复制代码

CODE

github.com/Bigerfe/fe-…

上一个栗子的多文件上传只有一个进度条,有些需求可能会不大一样,需要观察到每个文件的上传进度,并且可以终止上传。

DEMO

文件上传下载 - 图8

说明

  • 为了预览的需要,我们这里选择上传图片文件,其他类型的也一样,只是预览不方便
  • 页面内增加一个多图预览的容器div.img-box
  • 根据选择的文件信息动态创建所属的预览区域和进度条以及取消按钮
  • 为取消按钮绑定事件,调用xhr.abort();终止上传
  • 使用window.URL.createObjectURL预览图片,在图片加载成功后需要清除使用的内存window.URL.revokeObjectURL(this.src);

HTML

  1. <div>
  2. 选择文件(可多选):
  3. <div class="addfile">添加文件
  4. <input type="file" id="f1" multiple />
  5. </div>
  6. <div class="img-box"></div>
  7. <button type="button" id="btn-submit">上 传</button>
  8. </div>
  9. 复制代码

JS

  1. <script>
  2. //更改网络 为慢3g,就可以比较明显的看到进度条了
  3. var fileMaxCount=6;
  4. var imgBox =document.getElementsByClassName('img-box')[0];
  5. var willUploadFile=[];//保存待上传的文件以及相关附属信息
  6. document.getElementById('f1').addEventListener('change',function (e) {
  7. var fileList = document.getElementById('f1').files;
  8. if (willUploadFile.length > fileMaxCount || fileList.length>fileMaxCount || (willUploadFile.length+ fileList.length>fileMaxCount)) {
  9. alert('最多只能上传' + fileMaxCount + '张图');
  10. return;
  11. }
  12. for (var i = 0; i < fileList.length; i++) {
  13. var f = fileList[i];//先预览图片
  14. var img = document.createElement('img');
  15. var item = document.createElement('div');
  16. var progress = document.createElement('div');
  17. progress.className='progress';
  18. progress.innerHTML = '<span class="red"></span><button type="button">Abort</button>';
  19. item.className='item';
  20. img.src = window.URL.createObjectURL(f);
  21. img.onload = function () {
  22. //显示要是否这块儿内存
  23. window.URL.revokeObjectURL(this.src);
  24. }
  25. item.appendChild(img);
  26. item.appendChild(progress);
  27. imgBox.appendChild(item);
  28. willUploadFile.push({
  29. file:f,
  30. item,
  31. progress
  32. });
  33. }
  34. });
  35. function xhrSend({file, progress}) {
  36. var progressSpan = progress.firstElementChild;
  37. var btnCancel = progress.getElementsByTagName('button')[0];
  38. var abortFn=function(){
  39. if(xhr && xhr.readyState!==4){
  40. //取消上传
  41. xhr.abort();
  42. }
  43. }
  44. btnCancel.removeEventListener('click',abortFn);
  45. btnCancel.addEventListener('click',abortFn);
  46. progressSpan.style.width='0';
  47. progressSpan.classList.remove('green');
  48. var fd = new FormData(); //构造FormData对象
  49. fd.append('f1',file);
  50. var xhr = new XMLHttpRequest(); //创建对象
  51. xhr.open('POST', 'http://localhost:8100/', true);
  52. xhr.onreadystatechange = function () {
  53. console.log('state change', xhr.readyState);
  54. //调用 abort 后,state 立即变成了4,并不会变成0
  55. //增加自定义属性 xhr.uploaded
  56. if (xhr.readyState == 4 && xhr.uploaded) {
  57. var obj = JSON.parse(xhr.responseText); //返回值
  58. console.log(obj);
  59. if(obj.fileUrl.length){
  60. //alert('上传成功');
  61. }
  62. }
  63. }
  64. xhr.onprogress=updateProgress;
  65. xhr.upload.onprogress = updateProgress;
  66. function updateProgress(event) {
  67. if (event.lengthComputable) {
  68. var completedPercent = (event.loaded / event.total * 100).toFixed(2);
  69. progressSpan.style.width= completedPercent+'%';
  70. progressSpan.innerHTML=completedPercent+'%';
  71. if(completedPercent>90){//进度条变色
  72. progressSpan.classList.add('green');
  73. }
  74. if(completedPercent>=100){
  75. xhr.uploaded=true;
  76. }
  77. console.log('已上传',completedPercent);
  78. }
  79. }
  80. //注意 send 一定要写在最下面,否则 onprogress 只会执行最后一次 也就是100%的时候
  81. xhr.send(fd);//发送时 Content-Type默认就是: multipart/form-data;
  82. return xhr;
  83. }
  84. //文件上传
  85. function submitUpload(willFiles) {
  86. if(!willFiles.length){
  87. return;
  88. }
  89. //遍历文件信息进行上传
  90. willFiles.forEach(function (item) {
  91. xhrSend({
  92. file:item.file,
  93. progress:item.progress
  94. });
  95. });
  96. }
  97. //绑定提交事件
  98. document.getElementById('btn-submit').addEventListener('click',function () {
  99. submitUpload(willUploadFile);
  100. });
  101. </script>
  102. 复制代码

问题 1

这里没有做上传的并发控制,可以通过控制同时可上传文件的个数(这里控制为最多 6 个)或者上传的时候做好并发处理,也就是同时只能上传 X 个文件。

问题 2

在测试过程中,取消请求的方法xhr.abort()调用后,xhr.readyState会立即变为4,而不是0,所以这里需要做容错处理。

MDN 上说是 0.

文件上传下载 - 图9

如果大家有不同的结果,欢迎留言。

CODE

github.com/Bigerfe/fe-…

html5的出现,让拖拽上传交互成为可能,现在这样的体验也屡见不鲜。

DEMO

文件上传下载 - 图10

说明

  • 定义一个允许拖放文件的区域div.drop-box
  • 取消drop 事件的默认行为e.preventDefault();,不然浏览器会直接打开文件
  • 为拖拽区域绑定事件,鼠标在拖拽区域上 dragover, 鼠标离开拖拽区域dragleave, 在拖拽区域上释放文件drop
  • drop事件内获得文件信息e.dataTransfer.files

HTML

  1. <div class="drop-box" id="drop-box">
  2. 拖动文件到这里,开始上传
  3. </div>
  4. <button type="button" id="btn-submit">上 传</button>
  5. 复制代码

JS

  1. <script>
  2. var box = document.getElementById('drop-box');
  3. //禁用浏览器的拖放默认行为
  4. document.addEventListener('drop',function (e) {
  5. console.log('document drog');
  6. e.preventDefault();
  7. });
  8. //设置拖拽事件
  9. function openDropEvent() {
  10. box.addEventListener("dragover",function (e) {
  11. console.log('elemenet dragover');
  12. box.classList.add('over');
  13. e.preventDefault();
  14. });
  15. box.addEventListener("dragleave", function (e) {
  16. console.log('elemenet dragleave');
  17. box.classList.remove('over');
  18. e.preventDefault();
  19. });
  20. box.addEventListener("drop", function (e) {
  21. e.preventDefault(); //取消浏览器默认拖拽效果
  22. var fileList = e.dataTransfer.files; //获取拖拽中的文件对象
  23. var len=fileList.length;//用来获取文件的长度(其实是获得文件数量)
  24. //检测是否是拖拽文件到页面的操作
  25. if (!len) {
  26. box.classList.remove('over');
  27. return;
  28. }
  29. box.classList.add('over');
  30. window.willUploadFileList=fileList;
  31. }, false);
  32. }
  33. openDropEvent();
  34. function submitUpload() {
  35. var fileList = window.willUploadFileList||[];
  36. if(!fileList.length){
  37. alert('请选择文件');
  38. return;
  39. }
  40. var fd = new FormData(); //构造FormData对象
  41. for(var i =0;i<fileList.length;i++){
  42. fd.append('f1', fileList[i]);//支持多文件上传
  43. }
  44. var xhr = new XMLHttpRequest(); //创建对象
  45. xhr.open('POST', 'http://localhost:8100/', true);
  46. xhr.onreadystatechange = function () {
  47. if (xhr.readyState == 4) {
  48. var obj = JSON.parse(xhr.responseText); //返回值
  49. if(obj.fileUrl.length){
  50. alert('上传成功');
  51. }
  52. }
  53. }
  54. xhr.send(fd);//发送
  55. }
  56. //绑定提交事件
  57. document.getElementById('btn-submit').addEventListener('click',submitUpload);
  58. </script>
  59. 复制代码

CODE

github.com/Bigerfe/fe-…

掘金的写文编辑器是支持粘贴上传图片的,比如我从磁盘粘贴或者从网页上右键复制图片。

DEMO

文件上传下载 - 图11

说明

  • 页面内增加一个可编辑的编辑区域div.editor-box,开启contenteditable
  • div.editor-box绑定paste事件
  • 处理paste 事件,从event.clipboardData || window.clipboardData获得数据
  • 将数据转换为文件items[i].getAsFile()
  • 实现在编辑区域的光标处插入内容 insertNodeToEditor 方法

问题 1

测试中发现复制多个文件无效,只有最后一个文件上传,在掘金的编辑器里也同样存在,在坐有知道原因的可以留言说下。

问题 2

mac系统可以支持从磁盘复制文件后上传,windows 系统测试未通过,剪贴板的数据未拿到。

HTML

  1. <div class="editor-box" id="editor-box" contenteditable="true" >
  2. 可以直接粘贴图片到这里直接上传
  3. </div>
  4. 复制代码

JS

  1. //光标处插入 dom 节点
  2. function insertNodeToEditor(editor,ele) {
  3. //插入dom 节点
  4. var range;//记录光标位置对象
  5. var node = window.getSelection().anchorNode;
  6. // 这里判断是做是否有光标判断,因为弹出框默认是没有的
  7. if (node != null) {
  8. range = window.getSelection().getRangeAt(0);// 获取光标起始位置
  9. range.insertNode(ele);// 在光标位置插入该对象
  10. } else {
  11. editor.append(ele);
  12. }
  13. }
  14. var box = document.getElementById('editor-box');
  15. //绑定paste事件
  16. box.addEventListener('paste',function (event) {
  17. var data = (event.clipboardData || window.clipboardData);
  18. var items = data.items;
  19. var fileList = [];//存储文件数据
  20. if (items && items.length) {
  21. // 检索剪切板items
  22. for (var i = 0; i < items.length; i++) {
  23. console.log(items[i].getAsFile());
  24. fileList.push(items[i].getAsFile());
  25. }
  26. }
  27. window.willUploadFileList = fileList;
  28. event.preventDefault();//阻止默认行为
  29. submitUpload();
  30. });
  31. function submitUpload() {
  32. var fileList = window.willUploadFileList||[];
  33. var fd = new FormData(); //构造FormData对象
  34. for(var i =0;i<fileList.length;i++){
  35. fd.append('f1', fileList[i]);//支持多文件上传
  36. }
  37. var xhr = new XMLHttpRequest(); //创建对象
  38. xhr.open('POST', 'http://localhost:8100/', true);
  39. xhr.onreadystatechange = function () {
  40. if (xhr.readyState === 4) {
  41. var obj = JSON.parse(xhr.responseText); //返回值
  42. console.log(obj);
  43. if(obj.fileUrl.length){
  44. var img = document.createElement('img');
  45. img.src= obj.fileUrl[0];
  46. img.style.width='100px';
  47. insertNodeToEditor(box,img);
  48. // alert('上传成功');
  49. }
  50. }
  51. }
  52. xhr.send(fd);//发送
  53. }
  54. 复制代码

CODE

github.com/Bigerfe/fe-…

ie 时代由于无法使用xhr上传二进制数据,上传大文件需要借助浏览器插件来完成。 现在来看实现大文件上传简直 soeasy。

如果太大的文件,比如一个视频 1g 2g 那么大,直接采用上面的栗子中的方法上传可能会出链接现超时的情况,而且也会超过服务端允许上传文件的大小限制,所以解决这个问题我们可以将文件进行分片上传,每次只上传很小的一部分 比如 2M。

DEMO

文件上传下载 - 图12

文件上传下载 - 图13

说明

相信大家都对Blob 对象有所了解,它表示原始数据,也就是二进制数据,同时提供了对数据截取的方法slice,而 File 继承了Blob的功能,所以可以直接使用此方法对数据进行分段截图。

  • 把大文件进行分段 比如 2M,发送到服务器携带一个标志,暂时用当前的时间戳,用于标识一个完整的文件
  • 服务端保存各段文件
  • 浏览器端所有分片上传完成,发送给服务端一个合并文件的请求
  • 服务端根据文件标识、类型、各分片顺序进行文件合并
  • 删除分片文件

HTML

代码略,只需要一个 input file 标签。

JS

  1. //分片逻辑 像操作字符串一样
  2. var start=0,end=0;
  3. while (true) {
  4. end+=chunkSize;
  5. var blob = file.slice(start,end);
  6. start+=chunkSize;
  7. if(!blob.size){//截取的数据为空 则结束
  8. //拆分结束
  9. break;
  10. }
  11. chunks.push(blob);//保存分段数据
  12. }
  13. <script>
  14. function submitUpload() {
  15. var chunkSize=2*1024*1024;//分片大小 2M
  16. var file = document.getElementById('f1').files[0];
  17. var chunks=[], //保存分片数据
  18. token = (+ new Date()),//时间戳
  19. name =file.name,chunkCount=0,sendChunkCount=0;
  20. //拆分文件 像操作字符串一样
  21. if(file.size>chunkSize){
  22. //拆分文件
  23. var start=0,end=0;
  24. while (true) {
  25. end+=chunkSize;
  26. var blob = file.slice(start,end);
  27. start+=chunkSize;
  28. if(!blob.size){//截取的数据为空 则结束
  29. //拆分结束
  30. break;
  31. }
  32. chunks.push(blob);//保存分段数据
  33. }
  34. }else{
  35. chunks.push(file.slice(0));
  36. }
  37. chunkCount=chunks.length;//分片的个数
  38. //没有做并发限制,较大文件导致并发过多,tcp 链接被占光 ,需要做下并发控制,比如只有4个在请求在发送
  39. for(var i=0;i< chunkCount;i++){
  40. var fd = new FormData(); //构造FormData对象
  41. fd.append('token', token);
  42. fd.append('f1', chunks[i]);
  43. fd.append('index', i);
  44. xhrSend(fd,function () {
  45. sendChunkCount+=1;
  46. if(sendChunkCount===chunkCount){//上传完成,发送合并请求
  47. console.log('上传完成,发送合并请求');
  48. var formD = new FormData();
  49. formD.append('type','merge');
  50. formD.append('token',token);
  51. formD.append('chunkCount',chunkCount);
  52. formD.append('filename',name);
  53. xhrSend(formD);
  54. }
  55. });
  56. }
  57. }
  58. function xhrSend(fd,cb) {
  59. var xhr = new XMLHttpRequest(); //创建对象
  60. xhr.open('POST', 'http://localhost:8100/', true);
  61. xhr.onreadystatechange = function () {
  62. console.log('state change', xhr.readyState);
  63. if (xhr.readyState == 4) {
  64. console.log(xhr.responseText);
  65. cb && cb();
  66. }
  67. }
  68. xhr.send(fd);//发送
  69. }
  70. //绑定提交事件
  71. document.getElementById('btn-submit').addEventListener('click',submitUpload);
  72. </script>
  73. 复制代码

NODE

服务端需要做一些改动,保存分片文件、合并分段文件、删除分段文件。

PS

合并文件这里使用 stream pipe 实现,这样更节省内存,边读边写入,占用内存更小,效率更高,代码见fnMergeFile方法。

  1. //二次处理文件,修改名称
  2. app.use((ctx) => {
  3. var body = ctx.request.body;
  4. var files = ctx.request.files ? ctx.request.files.f1:[];//得到上传文件的数组
  5. var result=[];
  6. var fileToken = ctx.request.body.token;// 文件标识
  7. var fileIndex=ctx.request.body.index;//文件顺序
  8. if(files && !Array.isArray(files)){//单文件上传容错
  9. files=[files];
  10. }
  11. files && files.forEach(item=>{
  12. var path = item.path;
  13. var fname = item.name;//原文件名称
  14. var nextPath = path.slice(0, path.lastIndexOf('/') + 1) + fileIndex + '-' + fileToken;
  15. if (item.size > 0 && path) {
  16. //得到扩展名
  17. var extArr = fname.split('.');
  18. var ext = extArr[extArr.length - 1];
  19. //var nextPath = path + '.' + ext;
  20. //重命名文件
  21. fs.renameSync(path, nextPath);
  22. result.push(uploadHost+nextPath.slice(nextPath.lastIndexOf('/') + 1));
  23. }
  24. });
  25. if(body.type==='merge'){//合并分片文件
  26. var filename = body.filename,
  27. chunkCount = body.chunkCount,
  28. folder = path.resolve(__dirname, '../static/uploads')+'/';
  29. var writeStream = fs.createWriteStream(`${folder}${filename}`);
  30. var cindex=0;
  31. //合并文件
  32. function fnMergeFile(){
  33. var fname = `${folder}${cindex}-${fileToken}`;
  34. var readStream = fs.createReadStream(fname);
  35. readStream.pipe(writeStream, { end: false });
  36. readStream.on("end", function () {
  37. fs.unlink(fname, function (err) {
  38. if (err) {
  39. throw err;
  40. }
  41. });
  42. if (cindex+1 < chunkCount){
  43. cindex += 1;
  44. fnMergeFile();
  45. }
  46. });
  47. }
  48. fnMergeFile();
  49. ctx.body='merge ok 200';
  50. }
  51. });
  52. 复制代码

CODE

github.com/Bigerfe/fe-…

在上面我们实现了大文件的分片上传,解决了大文件上传超时和服务器的限制。

但是仍然不够完美,大文件上传并不是短时间内就上传完成,如果期间断网,页面刷新了仍然需要重头上传,这种时间的浪费怎么能忍?

所以我们实现断点续传,已上传的部分跳过,只传未上传的部分。

方法 1

在上面我们实现了文件分片上传和最终的合并,现在要做的就是如何检测这些分片,不再重新上传即可。 这里我们可以在本地进行保存已上传成功的分片,重新上传的时候使用spark-md5来生成文件 hash,区分此文件是否已上传。

  • 为每个分段生成 hash 值,使用 spark-md5
  • 将上传成功的分段信息保存到本地
  • 重新上传时,进行和本地分段 hash 值的对比,如果相同的话则跳过,继续下一个分段的上传

PS

生成 hash 过程肯定也会耗费资源,但是和重新上传相比可以忽略不计了。

DEMO

文件上传下载 - 图14

文件上传下载 - 图15

HTML

  1. 代码略
  2. 复制代码

JS

模拟分段保存,本地保存到localStorage

  1. //获得本地缓存的数据
  2. function getUploadedFromStorage(){
  3. return JSON.parse( localStorage.getItem(saveChunkKey) || "{}");
  4. }
  5. //写入缓存
  6. function setUploadedToStorage(index) {
  7. var obj = getUploadedFromStorage();
  8. obj[index]=true;
  9. localStorage.setItem(saveChunkKey, JSON.stringify(obj) );
  10. }
  11. //分段对比
  12. var uploadedInfo = getUploadedFromStorage();//获得已上传的分段信息
  13. for(var i=0;i< chunkCount;i++){
  14. console.log('index',i, uploadedInfo[i]?'已上传过':'未上传');
  15. if(uploadedInfo[i]){//对比分段
  16. sendChunkCount=i+1;//记录已上传的索引
  17. continue;//如果已上传则跳过
  18. }
  19. var fd = new FormData(); //构造FormData对象
  20. fd.append('token', token);
  21. fd.append('f1', chunks[i]);
  22. fd.append('index', i);
  23. (function (index) {
  24. xhrSend(fd, function () {
  25. sendChunkCount += 1;
  26. //将成功信息保存到本地
  27. setUploadedToStorage(index);
  28. if (sendChunkCount === chunkCount) {
  29. console.log('上传完成,发送合并请求');
  30. var formD = new FormData();
  31. formD.append('type', 'merge');
  32. formD.append('token', token);
  33. formD.append('chunkCount', chunkCount);
  34. formD.append('filename', name);
  35. xhrSend(formD);
  36. }
  37. });
  38. })(i);
  39. }
  40. 复制代码

方法 2

为什么还有方法 2 呢,正常情况下方法 1 没问题,但是需要将分片信息保存在客户端,保存在客户端是最不保险的,说不定出现各种神奇的幺蛾子。

所以这里有一个更完善的实现,只提供思路,代码就不写了,也是基于上面的实现,只是服务端需要增加一个接口。

基于上面一个栗子进行改进,服务端已保存了部分片段,客户端上传前需要从服务端获取已上传的分片信息(上面是保存在了本地浏览器),本地对比每个分片的 hash 值,跳过已上传的部分,只传未上传的分片。

方法 1 是从本地获取分片信息,这里只需要将此方法的能力改为从服务端获取分片信息就行了。

  1. -getUploadedFromStorage
  2. +getUploadedFromServer(fileHash)
  3. 复制代码

另外服务端增加一个获取分片的接口供客户端调用,思路最重要,代码就不贴了。

不只会从客户端上传文件到服务器,服务器也会上传文件到其他服务器。

  • 读取文件 buffer fs
  • 构建 form-data form-data
  • 上传文件 node-fetch

NODE

  1. /**
  2. * filepath = 相对根目录的路径即可
  3. */
  4. async function getFileBufer(filePath) => {
  5. return new Promise((resolve) => {
  6. fs.readFile(filePath, function (err, data) {
  7. var bufer = null;
  8. if (!err) {
  9. resolve({
  10. err: err,
  11. data: data
  12. });
  13. }
  14. });
  15. });
  16. }
  17. /**
  18. * 上传文件
  19. */
  20. let fetch = require('node-fetch');
  21. let formData = require('form-data');
  22. module.exports = async (options) => {
  23. let {
  24. imgPath
  25. } = options;
  26. let data = await getFileBufer(imgPath);
  27. if (data.err) {
  28. return null;
  29. }
  30. let form = new formData();
  31. form.append('xxx', xxx);
  32. form.append('pic', data.data);
  33. return fetch('http://xx.com/upload', {
  34. body: form,
  35. method: 'POST',
  36. headers: form.getHeaders()//要活的 form-data的头,否则无法上传
  37. }).then(res => {
  38. return res.json();
  39. }).then(data => {
  40. return data;
  41. })
  42. }
  43. 复制代码

在浏览器端对文件的类型、大小、尺寸进行判断

  • file.type判断类型
  • file.size判断大小
  • 通过动态创建 img 标签,图片加载后获得尺寸,naturalWidth naturalHeightor width height

JS

  1. var file = document.getElementById('f1').files[0];
  2. //判断类型
  3. if(f.type!=='image/jpeg' && f.type !== 'image/jpg' ){
  4. alert('只能上传 jpg 图片');
  5. flag=false;
  6. break;
  7. }
  8. //判断大小
  9. if(file.size>100*1024){
  10. alert('不能大于100kb');
  11. }
  12. //判断图片尺寸
  13. var img =new Image();
  14. img.onload=function(){
  15. console.log('图片原始大小 width*height', this.width, this.height);
  16. if(this.naturalWidth){
  17. console.log('图片原始大小 naturalWidth*naturalHeight', this.naturalWidth, this.naturalHeight);
  18. }else{
  19. console.log('oImg.width*height', this.width, this.height);
  20. }
  21. }
  22. 复制代码

input file 外观更改

由于 input file 的外观比较传统,很多地方都需要进行美化。

  1. 定义好一个外观,然后将 file input 定位到该元素上,让他的透明度为 0。
  2. 使用 label 标签
  1. <label for="file">Choose file to upload</label>
  2. <input type="file" id="file" name="file" multiple>
  3. 复制代码
  1. 隐藏 input file 标签,然后调用 input 元素的 click 方法

PS

file 标签隐藏后在 ie 下无法获得文件内容,建议还是方法1 兼容性强。

以上代码均已上传 github

github.com/Bigerfe/fe-…

developer.mozilla.org/zh-CN/docs/…

developer.mozilla.org/zh-CN/docs/…

cloud.tencent.com/developer/n…