当前位置: 首页 > article >正文

node实现大文件切片上传的方法

切片上传定义

文件切片上传,也称为分片上传,是一种处理大文件上传的有效方法。该方法通过将大文件分割成多个较小的部分(即切片或分片),然后分别上传这些切片到服务器,最后在服务器上将这些切片合并成原始文件

背景与需求

在Web开发中,上传大文件时常常会遇到各种问题,如连接超时、网络中断等,这些都可能导致上传失败。为了提升大文件上传的效率和成功率,文件切片上传技术应运而生。通过将大文件分割成多个小切片进行上传,可以有效减少单次上传的数据量,降低上传失败的风险,并提高上传速度。

工作原理

前端将用户上传的文件按照设定大小进行切片,每一个切片都会发送一次切片请求,等所有的切片请求都完成之后,再执行一次切换合并的请求,将各个切片进行拼接存储。

可拓展

1、切片的顺序可能是乱序的,需要在发送切片请求之前,先按照特定的规则进行排序,特定规则可以是从自定义的切片文件名入手
2、前端可以对切片上传进行监听,当某个切片上传失败时,需要触发重试机制上传该切片,可以设置重试次数或者重试时间间隔

优点与应用场景

提升上传效率:通过并行上传多个切片,可以显著提高大文件的上传速度
降低上传失败风险:即使某个切片上传失败,也只需要重新上传该切片,而不需要重新上传整个文件
支持断点续传:结合前端记录的上传进度和状态信息,可以实现断点续传功能,即在网络中断后可以从上次中断的位置继续上传

效果

启动本地服务器

 发送请求

切片数据

 

 最终合并的数据

代码

<body></body>
  <input type="file" id="file">
</body>

<script>
  const file = document.getElementById('file');
  file.addEventListener('change', (event)=>{
    const fileInfo = event.target.files[0]; // 获取文件信息
    console.log(fileInfo, 'fileInfo')
    const chunks = chunkFun(fileInfo) // 获取切片数组
    uploadFile(chunks, fileInfo.name) // 上传切片
  })

  const chunkFun = (fileInfo, size = 1024 * 1024 * 4) => { // size-自定义切片大小,这里是4M
    const chunks = [];
    for(let i=0; i<fileInfo.size; i+=size){
      chunks.push(fileInfo.slice(i, i + size))
    }
    return chunks
  }

  const uploadFile = (chunks, fileName) =>{
    const List = [];
    for(let i=0; i<chunks.length; i++){
      const formData = new FormData();
      formData.append('index', i);
      formData.append('total', chunks.length);
      formData.append('fileName', 'cheney');
      formData.append('file', chunks[i]);
      // 将每一个分片的fetch存在List数组中
      List.push(fetch('http://127.0.0.1:8080/up', {
        method: 'POST',
        body: formData
      }))
    }

    // 通过promise.all 并发发送请求
    Promise.all(List).then(res=>{
      fetch('http://127.0.0.1:8080/merge', {
        method: 'POST',
        headers: {
          'Content-Type': 'application/json'
        },
        body: JSON.stringify({
          fileName: fileName
        })
      }).then(res=>{
        console.log(res)
      })
    })
  }
</script>

node.js

// import express from 'express'
// import multer from 'multer'
// import cors from 'cors'
// import fs from 'node:fs'
// import path from 'node:path'
const express = require('express');
const multer = require('multer');
const cors = require('cors');
const fs = require('fs');
const path = require('path');


const storage = multer.diskStorage({
  destination: (req, file, cb) => {
      cb(null, 'chunk/uploads/') // 分片存储目录
  },
  filename: (req, file, cb) => {
    cb(null, `${req.body.index}-${req.body.fileName}`) // 分片文件名
  }
})
const upload = multer({ storage })
const app = express()

app.use(cors())
app.use(express.json())

app.post('/up', upload.single('file'), (req, res) => {
  res.send('ok')
})

app.post('/merge', async (req, res) => {
  const uploadPath = './chunk/uploads'
  let files = fs.readdirSync(path.join(process.cwd(), uploadPath)) // 获取所有的分片数据
  files = files.sort((a, b) => a.split('-')[0] - b.split('-')[0]) // 将分片按照文件名进行排序
  const writePath = path.join(process.cwd(), uploadPath, `${req.body.fileName}`) // 生成新的文件路径
  files.forEach((item) => {
      fs.appendFileSync(writePath, fs.readFileSync(path.join(process.cwd(), uploadPath, item))) // 读取分片信息,追加到新文件路径尾部
      fs.unlinkSync(path.join(process.cwd(), uploadPath, item)) // 将读取过的分片进行删除
  })

  res.send('ok')
})

app.listen(8080, () => {
  console.log('Server is running on port 8080')
})


http://www.kler.cn/news/325590.html

相关文章:

  • R整理数据技巧
  • 解决sortablejs+el-table表格内限制回撤和拖拽回撤失败问题
  • Redis string类型hash类型
  • 在pycharm中怎样调试HTML网页程序
  • B-树(不是B减树)原理剖析(1)
  • 有些硬盘录像机接入视频汇聚平台EasyCVR后通道不显示/显示不全,该如何处理?
  • Qt 学习第十一天:QTableWidget 的使用
  • 启动hadoop集群出现there is no HDFS_NAMENODE_USER defined.Aborting operation
  • 基于HTML、CSS和JavaScript的滚动数字显示效果,类似于老式计数器或电子表上的数字滚动效果
  • JAVA并发编程高级——JDK 新增的原子操作类 LongAdder
  • 渗透测试实战—教育攻防演练信息收集
  • Bugku 渗透测试1
  • 03. 前端面试题之ts : typescript 的数据类型有哪些?
  • LeetCode 热题 100 回顾2
  • 3种方法解决Docker容器中配置运行环境问题
  • 使用Python实现图形学的法线映射算法
  • 磁盘管理器
  • Qt网络编程——QUdpSocket
  • golang学习笔记24-文件操作
  • 实用好软-----电脑端 从视频中导出音频的方便工具
  • 微信小程序-数据模型与动态赋值
  • 女性向游戏的新战场:AI陪伴系统
  • CSS中的字体样式、文本样式、列表样式以及背景和渐变
  • VS Code、Git与自动化脚本的效能之旅
  • MATLAB GUI设计原则与实践
  • GPT带我学-设计模式18-访问者模式
  • Springboot项目-实战2-实现
  • 计数排序(counting sort)
  • 计算机毕业设计 饮食营养管理信息系统的设计与实现 Java实战项目 附源码+文档+视频讲解
  • 问题:vscode 打印中文时终端输出乱码