我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Vue如何实现封装一个切片上传组件

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Vue如何实现封装一个切片上传组件

今天小编给大家分享一下Vue如何实现封装一个切片上传组件的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。

    组件效果

    单文件切片上传

    Vue如何实现封装一个切片上传组件

    多文件切片上传

    Vue如何实现封装一个切片上传组件

    组件使用案例

    <template>  <div id="app">    <div class="upload-wrap">      <UploadSlice        :action="uploadInfoSlice.actionChunk"        :headers="uploadInfoSlice.headers"        :limit="uploadInfoSlice.limit"        :accept="uploadInfoSlice.accept"        :show-file-list="false"        cancelable        :on-success="handleSuccess"        :on-remove="handleRemove"        :on-cancel="handleCancel"        :on-upload-pre="handleUploadPre"        :on-upload-merge="handleUploadMerge"        :on-form-data="genFormData"      />    </div>  </div></template><script>import UploadSlice from './components/UploadSlice.vue'import { uploadPre, uploadMerge } from '@/api/upload-slice'export default {  name: 'App',  components: {    UploadSlice  },  data() {    return {      // 上传部分      uploadInfoSlice: {        actionChunk: process.env.VUE_APP_BASE_API + '/storage/file/v3/chunk', // 切片请求上传路径        headers: { 'Authorization': 'Bearer XXX' }      }    }  },  methods: {    // 分片预请求    async handleUploadPre(file) {      const form = new FormData()      form.append('fileSource', 'APPLICATION')      form.append('originFileName', file.name)      let res = ''      try {        res = await uploadPre(form)      } catch (error) {        throw new Error(error)      }    },    // 构造分片参数    genFormData(chunks, uid) {      const prepareId = this.getCurrentPrepareId(uid)      return chunks.map(chunk => {        const form = new FormData()        form.append('chunk', chunk.file)        form.append('uploadId', prepareId)        form.append('partNumber', chunk.index)        return form      })    },    // 合并请求    async handleUploadMerge(file, uid) {      const prepareId = this.getCurrentPrepareId(uid)      const form = new FormData()      form.append('fileSource', 'APPLICATION')      form.append('hash', prepareId)      form.append('filename', file.name)      // return 建议使用, 用于handleSuccess获取数据      try {        const res = await uploadMerge(form)        return res      } catch (error) {        return error      }    },    // 判断当前处理prepareId    getCurrentPrepareId(uid) {      for (const item of this.progressFileList) {        if (item.uid === uid) {          return item.prepareId        }      }    }  }}</script>

    使用文档

    Attribute

    标红色部分为二次封装处理过的功能,其他为el-upload自带属性

    参数说明类型可选值默认值备注
    action必选参数,分片上传的地址,预请求和合并请求在组件外操作String--
    headers设置上传的请求头部String--
    multiple是否支持多选文件boolean-

    accept可上传文件类型,多种类型用","分隔 (格式不符合自动提示)String--
    on-remove文件列表移除文件时的钩子function(file, fileList)&mdash;&mdash;
    on-success文件上传成功时的钩子function(response, file, fileList)&mdash;&mdash;
    on-error文件上传失败时的钩子function(err, file, fileList)&mdash;&mdash;
    on-progress文件上传时的钩子function(event, file, fileList)&mdash;&mdash;
    on-change文件状态改变时的钩子,添加文件、上传成功和上传失败时都会被调用function(file, fileList)&mdash;&mdash;
    on-exceed文件超出个数限制时的钩子function(files, fileList)&mdash;&mdash;
    list-type文件列表的类型stringtext/picture/picture-cardtext
    show-file-list是否显示已上传文件列表(文件分片上传时建议设置false,否则会有两个进度条)boolean&mdash;true
    file-list上传的文件列表, 例如: [{name: 'food.jpg', url: 'xxx.cdn.com/xxx.jpg'}]array&mdash;[]
    disabled是否禁用boolean&mdash;false
    cancelable是否支持取消boolean&mdash;false
    limit最多允许上传个数(超出个数自动提示)number&mdash;&mdash;
    size限制大小String&mdash;&mdash;
    hideBtn是否在上传过程中隐藏上传按钮boolean&mdash;&mdash;false

    Slot

    插槽名说明
    trigger触发文件选择框的内容
    tip提示说明文字
    more-tips在默认提示后补充说明

    封装过程

    切片上传组件是基于el-upload进行的二次封装,文章开头组件效果演示可以看到上传一个文件会发送三个请求:prepare,chunk, merge,也就是整个上传过程,主要分为三步:1.预请求 2.分片请求 3.合并请求,预请求和合并请求就是我们正常的http请求,主要处理的是分片请求,分片请求主要的步骤是:

    • 将文件切片

    • 构造切片请求参数

    • 控制分片请求的并发

    1. 文件切片

    el-upload上传后, 在on-change属性的回调里可以获取文件file,通过file.raw.slice对文件进行切片,目前的切片规则是:1.小于10M 固定一片 2.小于50M 文件10%为一片 3.大于50M 固定5M 一片(可以根据自己的需求进行修改)

    genFileChunks(file) {  const chunks = []  let cur = 0  // 小于10M 固定一片  if (file.size < (10 * 1024 * 1024)) {    chunks.push({      index: cur,      file: file.raw.slice(cur, file.size),      originFilename: file.name    })    return chunks  }  // 小于50M 文件10%为一片  if (file.size < (50 * 1024 * 1024)) {    const chunkSize = parseInt(file.size * 0.1)    while (cur < file.size) {      chunks.push({        index: cur,        file: file.raw.slice(cur, cur + chunkSize),        originFilename: file.name      })      cur += chunkSize    }    return chunks  }  // 大于50M 固定5M 一片  const chunkSize = parseInt(5 * 1024 * 1024)  while (cur < file.size) {    chunks.push({      index: cur,      file: file.raw.slice(cur, cur + chunkSize),      originFilename: file.name    })    cur += chunkSize  }  return chunks},

    一个32M的文件按照10%切一片,构造好的切片数据是这样的

    Vue如何实现封装一个切片上传组件

    2. 构造切片请求参数

    切片请求不同业务的参数是变化的,所以参数部分可以抛出给父组件处理,增加组件的复用性

    父组件

    <template>  <UploadSlice    :action="uploadInfoSlice.actionChunk"    :headers="uploadInfoSlice.headers"    :on-form-data="genFormData"  /></template><script>  methods: {    // 构造分片参数    genFormData(chunks, uid) {      const prepareId = this.getCurrentPrepareId(uid)      return chunks.map(chunk => {        const form = new FormData()        form.append('chunk', chunk.file)        form.append('uploadId', prepareId)        form.append('partNumber', chunk.index)        return form      })    },  },</script>

    子组件

    <template>  <el-upload      action=""      :accept="accept"  ></template><script>  props: {    onFormData: {      type: Function,      default: () => {}    },  },  methods: {    async uploadChunks(uid) {      // 预请求      // ---------------      // 上传切片      const requests = this._genRequest(this._genUploadData(uid), uid)      // 控制并发      await this.sendRequest(requests)            // 合并请求      // ---------------    },        // 构造分片参数    _genUploadData(uid) {      const chunks = this.getCurrentChunks(uid)      return this.onFormData(chunks, uid)    },        // 生成调用请求:[Promise, Promise]    _genRequest(uploadData, uid) {      console.log('uploadData', uploadData)      const file = this.getCurrentFile(uid)      const chunks = this.getCurrentChunks(uid)      return uploadData.map((form, index) => {        const options = {          headers: this.$attrs.headers,          file: file,          data: form,          action: this.action,          onProgress: progress => {            chunks[index].progress = Number(              ((progress.loaded / progress.total) * 100).toFixed(2)            )            this.handleProgress(progress, file, uid)          }        }        return options      })    },  },</script>

    3. 控制分片请求的并发

    切片上传如果不控制并发,在分片很多时,就会同时发送很多个http请求,导致线程阻塞,影响页面其他请求的操作,所以控制并发是需要的。我设置的是最多允许3个并发请求。

        sendRequest(requests, limit = 3) {      return new Promise((resolve, reject) => {        const len = requests.length        let counter = 0        let isTips = false // 只提示一次失败        let isStop = false // 如果一个片段失败超过三次 认为当前网洛有问题 停止全部上传        const startRequest = async() => {          if (isStop) return          const task = requests.shift()          if (task && task.file.status !== 'cancel') {            // 利用try...catch捕获错误            try {              // 具体的接口  抽离出去了              await ajax(task)              if (counter === len - 1) { // 最后一个任务                resolve()              } else { // 否则接着执行                counter++                startRequest() // 启动下一个任务              }            } catch (error) {              // 网络异常              if (error === 'NETWORK_ERROR' && !isTips) {                Message.error('网络异常,文件上传失败')                this.upLoading = false                this.preLoading = false                isTips = true                this.handleRemove('', [])              }              // 接口报错重试,限制为3次              if (task.error < 3) {                task.error++                requests.unshift(task)                startRequest()              } else {                isStop = true                reject(error)              }            }          }        }        // 启动任务        while (limit > 0) {          // 模拟不同大小启动          setTimeout(() => {            startRequest()          }, Math.random() * 2000)          limit--        }      })    }  }

    以上就是“Vue如何实现封装一个切片上传组件”这篇文章的所有内容,感谢各位的阅读!相信大家阅读完这篇文章都有很大的收获,小编每天都会为大家更新不同的知识,如果还想学习更多的知识,请关注编程网行业资讯频道。

    免责声明:

    ① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

    ② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

    Vue如何实现封装一个切片上传组件

    下载Word文档到电脑,方便收藏和打印~

    下载Word文档

    猜你喜欢

    Vue如何实现封装一个切片上传组件

    今天小编给大家分享一下Vue如何实现封装一个切片上传组件的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。组件效果单文件切片上传
    2023-07-05

    Vue实现封装一个切片上传组件

    平时业务开发中用el-upload能满足大部分场景,但是对于一些大文件的上传时会比较慢,所以自己基于el-upload封装了一个切片上传组件,希望对大家有所帮助
    2023-03-19

    vue-cropper插件如何实现图片截取上传组件封装

    小编给大家分享一下vue-cropper插件如何实现图片截取上传组件封装,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!需求场景:后台开发需要上传图片并进行相应比例尺寸图片的截取,本组件开发采用Ant Design Vue组
    2023-06-15

    vue-cropper组件如何实现图片切割上传

    小编给大家分享一下vue-cropper组件如何实现图片切割上传,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!vue-cropper在vue中的引入1、组件内引入
    2023-06-15

    vue如何实现文件切片上传

    这篇文章主要介绍了vue如何实现文件切片上传的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇vue如何实现文件切片上传文章都会有所收获,下面我们一起来看看吧。在实际开发项目过程中有时候需要上传比较大的文件,然后呢
    2023-07-05

    vue如何实现上传图片组件

    这篇文章给大家分享的是有关vue如何实现上传图片组件的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。未上传状态上传状态其他状态(查看/删除)自定义组件文件名称 - 这里叫UploadImg.vue
    2023-06-25

    vue如何实现上传组件

    这篇文章主要介绍vue如何实现上传组件,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!思路文件上传的两种实现方式1.From形式
    2023-06-15

    如何开发一个封装iframe的vue组件

    这篇文章给大家分享的是有关如何开发一个封装iframe的vue组件的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。VUE的基本组成单元,我看应该是组件。用VUE开发前端项目,就是开发一个个组件,然后搭积木一样,将项
    2023-06-14

    vue如何实现上传图片文件

    这篇文章给大家分享的是有关vue如何实现上传图片文件的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。原始input标签form表单上传
    2023-06-15

    基于Vue实现封装一个虚拟列表组件

    正常情况下,我们对于数据都会分页加载,最近项目中确实遇到了不能分页的场景,如果不分页,页面渲染几千条数据就会感知到卡顿,使用虚拟列表就势在必行了。本文主要介绍了如何基于Vue实现封装一个虚拟列表组件,感兴趣的可以了解一下
    2023-03-07

    基于Vue如何封装一个虚拟列表组件

    今天小编给大家分享一下基于Vue如何封装一个虚拟列表组件的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。组件效果使用方法
    2023-07-05

    如何使用vue封装一个自定义日历组件

    本文小编为大家详细介绍“如何使用vue封装一个自定义日历组件”,内容详细,步骤清晰,细节处理妥当,希望这篇“如何使用vue封装一个自定义日历组件”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。核心代码实现1、梳理思
    2023-07-05

    深析如何封装一个vue自定义日历组件

    本篇文章给大家带来了关于前端日历的相关知识,其中主要跟大家聊一聊如何封装一个自定义日历组件,感兴趣的朋友下面一起来看一下吧,希望对大家有帮助。
    2023-05-14

    php 如何一次实现多个照片上传

    这篇文章主要介绍了php 如何一次实现多个照片上传,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。php一次实现多个照片上传的方法:1、创建index.html文件;2、创建连
    2023-06-22

    编程热搜

    • Python 学习之路 - Python
      一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
      Python 学习之路 - Python
    • chatgpt的中文全称是什么
      chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
      chatgpt的中文全称是什么
    • C/C++中extern函数使用详解
    • C/C++可变参数的使用
      可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
      C/C++可变参数的使用
    • css样式文件该放在哪里
    • php中数组下标必须是连续的吗
    • Python 3 教程
      Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
      Python 3 教程
    • Python pip包管理
      一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
      Python pip包管理
    • ubuntu如何重新编译内核
    • 改善Java代码之慎用java动态编译

    目录