Vue小玩意儿:vue3+express.js实现大文件分片上传

vue3:

javascript 复制代码
<template>
  <div>
    <h1>大文件分片上传</h1>
    <input type="file" @change="onFileChange"/>
    <div v-if="progress > 0">
      上传进度: {{ progress }}%
    </div>
  </div>
</template>

<script setup>
import axios from 'axios'
import {ref} from 'vue'

const file = ref(null)
const progress = ref(0)

const onFileChange = (event) => {
  file.value = event.target.files[0]
  if (file.value) {
    uploadFile()
  }
}

const uploadFile = async () => {
  progress.value = 0

  const chunkSize = 1 * 1024 * 1024 // 1MB 分片
  const totalChunks = Math.ceil(file.value.size / chunkSize)

  for (let chunkIndex = 0; chunkIndex < totalChunks; chunkIndex++) {
    const start = chunkIndex * chunkSize
    const end = Math.min(start + chunkSize, file.value.size)
    const chunk = file.value.slice(start, end)

    const formData = new FormData()
    formData.append('file', chunk)
    formData.append('chunkIndex', chunkIndex)
    formData.append('totalChunks', totalChunks)
    formData.append('fileName', file.value.name)

    await axios.post('http://localhost:3000/upload', formData, {
      headers: {
        'Content-Type': 'multipart/form-data'
      }
    })

    progress.value = Math.round(((chunkIndex + 1) / totalChunks) * 100)
  }

  // await axios.post('http://localhost:3000/merge', {
  //   filename: this.selectedFile.name,
  //   totalChunks,
  // });

  alert('文件上传成功!');
}
</script>

express.js:

javascript 复制代码
const express = require('express');
const multer = require('multer');
const fs = require('fs-extra');
const path = require('path');
const cors = require('cors');

const app = express();
const PORT = 3000;

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

const upload = multer({dest: 'uploads/'});

app.post('/upload', upload.single('file'), async (req, res) => {
    const {originalname} = req.file;
    const {chunkIndex, totalChunks, fileName} = req.body;
    const chunkDir = path.join(__dirname, 'uploads', fileName.split('.')[0] + '_' + fileName.split('.')[1]);

    try {
        await fs.ensureDir(chunkDir);
        const chunkPath = path.join(chunkDir, chunkIndex);
        await fs.move(req.file.path, chunkPath);

        if (parseInt(chunkIndex) + 1 == parseInt(totalChunks)) {
            await mergeChunks(chunkDir, fileName);
            res.send({message: `Chunk ${chunkIndex} uploaded.Upload complete.`});
        } else {
            res.send({message: `Chunk ${chunkIndex} uploaded.`});
        }
    } catch (err) {
        console.error(err);
        res.status(500).send({message: 'Upload failed.', error: err.message});
    }
});

async function mergeChunks(chunkDir, fileName) {
    const filePath = path.join(__dirname, 'uploads', fileName);
    const chunkPaths = await fs.readdir(chunkDir);

    chunkPaths.sort((a, b) => parseInt(a) - parseInt(b));

    for (let chunkPath of chunkPaths) {
        const chunkFullPath = path.join(chunkDir, chunkPath);
        await fs.appendFile(filePath, await fs.readFile(chunkFullPath));
    }

    await fs.remove(chunkDir);
}

app.listen(PORT, () => {
    console.log(`Server is running on http://localhost:${PORT}`);
});

后端(Express.js):处理文件上传、分片存储和文件合并。

前端(Vue 3 + Axios):实现文件分片上传,并与后端进行通信。

分片上传:文件被切分为多个小块上传,后端在接收到所有分片后合并为完整的文件。

相关推荐
q***38512 小时前
SpringBoot + vue 管理系统
vue.js·spring boot·后端
喵个咪3 小时前
go-kratos-admin 快速上手指南:从环境搭建到启动服务(Windows/macOS/Linux 通用)
vue.js·go
用户841794814563 小时前
vxe-gantt table 甘特图如何设置任务视图每一行的背景色
vue.js
小章鱼学前端3 小时前
2025 年最新 Fabric.js 实战:一个完整可上线的图片选区标注组件(含全部源码).
前端·vue.js
涔溪3 小时前
实现将 Vue3 项目作为子应用,通过无界(Wujie)微前端框架接入到 Vue2 主应用中(Vue2 为主应用,Vue3 为子应用)
vue.js·前端框架·wujie
源码技术栈6 小时前
什么是云门诊系统、云诊所系统?
java·vue.js·spring boot·源码·门诊·云门诊
lcc1876 小时前
Vue3 ref函数和reactive函数
前端·vue.js
艾小码6 小时前
还在为组件通信头疼?defineExpose让你彻底告别传值烦恼
前端·javascript·vue.js
带只拖鞋去流浪6 小时前
迎接2026,重新认识Vue CLI (v5.x)
前端·vue.js·webpack
Coder-coco6 小时前
游戏助手|游戏攻略|基于SprinBoot+vue的游戏攻略系统小程序(源码+数据库+文档)
java·vue.js·spring boot·游戏·小程序·论文·游戏助手