React前端开发中实现断点续传

在前端开发中,断点续传是指在上传文件时,如果上传过程中中断(如网络断开、页面刷新等),可以从断点处继续上传,而不是重新上传整个文件。这种功能在大文件上传场景中非常有用。
以下是实现断点续传的思路和具体步骤:

实现思路

文件分片:

将大文件分割成多个小文件块(chunk)。
每个文件块单独上传。

记录上传进度:

使用本地存储(如 localStorage)或服务端记录已上传的文件块。

断点续传:

在上传前检查已上传的文件块,跳过已上传的部分。
从未上传的部分开始继续上传。

合并文件:

所有文件块上传完成后,通知服务端合并文件。

实现步骤

1. 前端实现

以下是基于 React 和 Axios 的断点续传实现示例:

import React, { useState, useRef } from "react";
import axios from "axios";const chunkSize = 1024 * 1024; // 每个文件块的大小(1MB)const FileUpload = () => {const [file, setFile] = useState(null);const [progress, setProgress] = useState(0);const fileInputRef = useRef(null);// 处理文件选择const handleFileChange = (e) => {const selectedFile = e.target.files[0];setFile(selectedFile);};// 上传文件const handleUpload = async () => {if (!file) return;const totalChunks = Math.ceil(file.size / chunkSize); // 总文件块数let uploadedChunks = 0; // 已上传的文件块数// 检查已上传的文件块const uploadedChunksFromStorage = JSON.parse(localStorage.getItem(file.name) || []);for (let i = 0; i < totalChunks; i++) {// 如果当前文件块已上传,跳过if (uploadedChunksFromStorage.includes(i)) continue;const start = i * chunkSize;const end = Math.min(start + chunkSize, file.size);const chunk = file.slice(start, end); // 获取文件块const formData = new FormData();formData.append("file", chunk);formData.append("fileName", file.name);formData.append("chunkIndex", i);formData.append("totalChunks", totalChunks);try {await axios.post("/upload", formData, {headers: { "Content-Type": "multipart/form-data" },});// 更新已上传的文件块uploadedChunksFromStorage.push(i);localStorage.setItem(file.name, JSON.stringify(uploadedChunksFromStorage));// 更新上传进度uploadedChunks++;setProgress(Math.round((uploadedChunks / totalChunks) * 100));} catch (error) {console.error("上传失败", error);return;}}// 所有文件块上传完成后,通知服务端合并文件try {await axios.post("/merge", { fileName: file.name });alert("上传完成");localStorage.removeItem(file.name); // 清除本地存储的上传记录setProgress(0);fileInputRef.current.value = ""; // 清空文件输入} catch (error) {console.error("文件合并失败", error);}};return (<div><h1>文件断点续传</h1><input type="file" onChange={handleFileChange} ref={fileInputRef} /><button onClick={handleUpload}>上传</button><div>上传进度: {progress}%</div></div>);
};export default FileUpload;

2. 服务端实现

服务端需要支持文件块的上传和合并。以下是一个简单的 Node.js 实现示例:

const express = require("express");
const multer = require("multer");
const fs = require("fs");
const path = require("path");const app = express();
const upload = multer({ dest: "uploads/" });// 上传文件块
app.post("/upload", upload.single("file"), (req, res) => {const { fileName, chunkIndex } = req.body;const chunkPath = path.join("uploads", `${fileName}-${chunkIndex}`);// 将文件块保存到临时目录fs.renameSync(req.file.path, chunkPath);res.send({ message: "文件块上传成功" });
});// 合并文件块
app.post("/merge", (req, res) => {const { fileName } = req.body;const chunksDir = path.join("uploads");const chunks = fs.readdirSync(chunksDir).filter((file) => file.startsWith(fileName)).sort((a, b) => a.split("-")[1] - b.split("-")[1]);// 创建可写流const writeStream = fs.createWriteStream(path.join("uploads", fileName));// 合并文件块chunks.forEach((chunk) => {const chunkPath = path.join(chunksDir, chunk);const readStream = fs.createReadStream(chunkPath);readStream.pipe(writeStream, { end: false });readStream.on("end", () => {fs.unlinkSync(chunkPath); // 删除已合并的文件块});});writeStream.on("finish", () => {res.send({ message: "文件合并成功" });});
});app.listen(3000, () => {console.log("服务端运行在 http://localhost:3000");
});

关键点解析

文件分片:

使用 File.slice() 方法将文件分割成多个块。
每个块单独上传。

记录上传进度:

使用 localStorage 记录已上传的文件块索引。
上传前检查已上传的文件块,跳过已上传的部分。

断点续传:

如果上传中断,重新上传时从未上传的文件块开始。

合并文件:

所有文件块上传完成后,通知服务端合并文件。
服务端按顺序读取文件块并写入目标文件。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/72576.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring 框架中常用注解和使用方法

Spring 框架中常用注解的详细解释与应用场景&#xff0c;结合核心功能和实际开发需求进行分类说明&#xff1a; 1.组件定义注解 1.1 Component 作用&#xff1a;通用注解&#xff0c;将普通 Java 类标记为 Spring 管理的 Bean&#xff0c;由容器实例化和管理&#xff0c;相当…

使用 PaddlePaddle 官方提供的 Docker 镜像

CUDA版本高PaddlePaddle不支持时&#xff0c;可以使用 PaddlePaddle 官方提供的 Docker 镜像 1. 安装 Docker Desktop1.1 下载 Docker Desktop1.2 安装 Docker Desktop1.3 启用 WSL 2 或 Hyper-V1.4 启动 Docker Desktop1.5 Docker不运行解决方法 2. 拉取 PaddlePaddle Docker …

VSCode C/C++ 环境搭建指南

VSCode C/C 环境搭建指南 第一步&#xff1a;下载并安装VSCode 访问官方网站 打开浏览器&#xff0c;访问Visual Studio Code官网。 选择平台 根据您的操作系统&#xff08;Windows、Mac、Linux&#xff09;选择对应的版本进行下载。 下载和安装 下载完成后&#xff0c;双击…

项目中使用柯里化函数

在项目中使用**柯里化函数&#xff08;Currying&#xff09;**可以显著提高代码的灵活性、可复用性和可读性。柯里化是一种将多参数函数转换为一系列单参数函数的技术。通过柯里化&#xff0c;可以将函数的调用方式从一次性传递所有参数改为分步传递参数&#xff0c;从而实现部…

Golang Channel 使用详解、注意事项与死锁分析

#作者&#xff1a;西门吹雪 文章目录 一、引言&#xff1a;Channel 在 Go 并发编程中的关键地位二、Channel 基础概念深度剖析2.1 独特特性2.2 类型与分类细解 三、Channel 基本使用实操指南3.1 声明与初始化3.3 单向 Channel 的运用 四、Channel 典型使用场景实战案例4.1 协程…

C语言经典代码题

1.输入一个4位数&#xff1a;输出这个输的个位 十位 百位 千位 #include <stdio.h> int main(int argc, char const *argv[]) {int a;printf("输入一个&#xff14;位数&#xff1a;");scanf("%d",&a);printf("个位&#xff1a;%d\n"…

stable-diffusion-webui-docker 构建 comfy-ui

Ubuntu 安装 stable-diffusion-webui-docker 常见问题处理方法 这篇文章介绍了在 Ubuntu 上安装 stable-diffusion-webui-docker&#xff0c;运行 docker compose --profile auto up --build 构建出的界面是 stable-diffusion-webui&#xff0c;如果运行 docker compose --prof…

【AI学习从零至壹】Pytorch神经⽹络

Pytorch神经⽹络 神经网络简介神经元激活函数 神经网络神经⽹络的⼯作过程前向传播(forward) 反向传播(backward)训练神经⽹络 Pytorch搭建并训练神经⽹络神经⽹络构建和训练过程数据预处理构建模型优化器&提取训练数据训练样本 神经网络简介 神经元 在深度学习中&#x…

stm32 L432KC(mbed)入门第一课

目录 一. 前言 二. 专栏意义 三. MS入门第一课 一. 前言 新的一年MS课程又开始了&#xff0c;同时也到了该专栏的第三个年头。在前两年中&#xff0c;该专栏帮助了很多第一次接触单片机的同学。其中&#xff0c;有的同学订阅专栏是为了更好的完成并且通过MS这门课程&#xf…

如何创建HTML自定义元素:使用 Web Component 的最佳实践

什么是 Web Component&#xff1f; Web Component 是一组允许开发者创建可复用、自定义 HTML 元素的技术。它们使得我们可以像原生 HTML 标签一样使用这些自定义元素&#xff0c;从而提升代码的模块化和复用性。Web Component 的核心技术有以下三部分&#xff1a; Custom Ele…

【系统架构设计师】操作系统 - 文件管理 ② ( 位示图 | 空闲区域 管理 | 位号 | 字号 )

文章目录 一、空闲区域 管理1、空闲区域分配2、空闲区域 管理方式 简介 二、位示图 简介1、位示图 表示2、位示图 字号3、位示图 位号4、位示图 中 比特位 分组管理 三、位示图 考点1、计算磁盘 位示图 的大小2、位示图 位置计算 一、空闲区域 管理 1、空闲区域分配 在 索引文件…

基于 Docker 和 Flask 构建高并发微服务架构

基于 Docker 和 Flask 构建高并发微服务架构 一、微服务架构概述 &#xff08;一&#xff09;微服务架构的优点 微服务架构是一种将应用程序拆分为多个小型、自治服务的架构风格&#xff0c;在当今的软件开发领域具有显著的优势。 高度可扩展性&#xff1a;每个微服务可以独…

搭建Django开发环境

搭建Django开发环境 文章目录 搭建Django开发环境[toc]一、安装Python语言环境二、安装Visual Studio Code三、安装setuptools工具四、安装Django框架 一、安装Python语言环境 1.测试当前系统环境是否存在Python语言解释器 python --version2.打开PowerShell终端&#xff0c;…

图论part3|101.孤岛的总面积、沉没孤岛、417. 太平洋大西洋水流问题

101. 孤岛的总面积 &#x1f517;&#xff1a;101. 孤岛的总面积思路&#xff1a;和昨天的岛的区别是&#xff1a;是否有挨着边的岛屿 所以可以先遍历四条边挨着的岛屿&#xff0c;把他们标记为非孤岛再计算其他岛屿当中的最大面积 代码&#xff1a;&#xff08;深度搜索&…

AP AR

混淆矩阵 真实值正例真实值负例预测值正例TPFP预测值负例FNTN &#xff08;根据阈值预测&#xff09; P精确度计算&#xff1a;TP/(TPFP) R召回率计算&#xff1a;TP/(TPFN) AP 综合考虑P R 根据不同的阈值计算出不同的PR组合&#xff0c; 画出PR曲线&#xff0c;计算曲线…

Ubuntu上部署Flask+MySQL项目

一、服务器安装python环境 1、安装gcc&#xff08;Ubuntu默认已安装&#xff09; 2、安装python源码 wget https://www.python.org/ftp/python/3.13.2/Python-3.13.2.tar.xz 3、安装Python依赖库 4、配置python豆瓣源 二、服务器安装虚拟环境 1、安装virtualenv pip3.10 ins…

深度学习有哪些算法?

深度学习包含多种算法和模型&#xff0c;广泛应用于图像处理、自然语言处理、语音识别等领域。以下是主要分类及代表性算法&#xff1a; 一、基础神经网络 多层感知机&#xff08;MLP&#xff09; 最简单的深度学习模型&#xff0c;由多个全连接层组成&#xff0c;用于分类和回…

【css酷炫效果】纯CSS实现按钮流光边框

【css酷炫效果】纯CSS实现按钮流光边框 缘创作背景html结构css样式完整代码效果图 【css酷炫效果】纯CSS实现按钮流光边框。 想直接拿走的老板&#xff0c;链接放在这里&#xff1a;https://download.csdn.net/download/u011561335/90490501 缘 创作随缘&#xff0c;不定时更…

【Android】ListView控件在进入|退出小窗下的异常

1&#xff0c;描述 页面使用了ListView控件&#xff0c;随后进入小窗模式&#xff0c;导致视图遮挡 2&#xff0c;根源 ListView虽然进入小窗relayout&#xff0c;其measureChild高度比全屏下要小&#xff0c;但是&#xff0c;其内部使用了Recycler机制&#xff0c;缓存了ite…

基于ssm的电子病历系统(全套)

一、系统架构 前端&#xff1a;jsp | bootstrap | jquery 后端&#xff1a;spring | springmvc | mybatis 环境&#xff1a;jdk1.8 | mysql | maven | tomcat | idea 二、代码及数据库 三、功能介绍 01. 登录 02. 主页 03. 管理员-个人中心-修改密码…