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

node.js 实战——express图片保存到本地或服务器(七牛云、腾讯云、阿里云)

本地

✅ 使用formidable 读取表单内容

npm i formidable 

✅ 使用mime-types 获取图片后缀

npm install mime-types

✅ js 中提交form表单

 document.getElementById('uploadForm').addEventListener('submit', function(e){e.preventDefault();const blob =preview._blob;if(!blob){alert("请先选择并裁剪头像");return;}const formData = new FormData();formData.append('file', blob);//未裁剪使用代码// const file =input.files[0];// if(!file) return alert('Please upload a valid image!');//// let formData = new FormData();// formData.append('file', file);let xhr = new XMLHttpRequest();xhr.open('POST', '/users/upload-avatar');xhr.upload.addEventListener('progress', (e) => {if(e.lengthComputable){let percent = Math.round((e.loaded / e.total) * 100);progressWrapper.classList.remove('d-none');progressBar.style.width= percent + '%';progressBar.innerText = percent + '%';}})xhr.onload =function () {if(xhr.status === 200) {alert("successfully uploaded!");}else{alert("fail to upload");}}xhr.send(formData);})

✅ express 中routers的js

var express = require('express');
var router = express.Router();
const { IncomingForm } = require('formidable'); // ✅ 注意不是直接 require('formidable')
const path = require('path');
const mime = require('mime-types')/* GET users listing. */
router.get('/', function(req, res, next) {res.render('user', { title: '订餐系统' });
});router.post('/upload-avatar', function(req, res, next) {//创建表单对象const form =new IncomingForm({multiple:true,//设置上传文件的保存目录uploadDir:__dirname+'/../public/uploads/',//保存文件后缀keepExtension:true,// 支持多文件multiples:true,filename:(name,ext,part,form)=>{const timestamp = Date.now();let extension = mime.extension(part.mimetype) || '';return `${timestamp}-${Math.random().toString(36).slice(2)}${extension ? '.' + extension : ''}`;}});//form.parse(req,function(err,fields,files){if(err){next(err);return;}res.json({files,files});})})module.exports = router;

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

七牛云

✅ 实现步骤概览

  1. 使用 formidable 获取上传的图片流或临时文件路径
  2. 使用 qiniu SDK 上传到七牛云对象存储
  3. 返回文件 URL 给前端

✅ 安装依赖

npm install formidable qiniu

✅ 七牛云配置准备

你需要准备这些信息:
• accessKey
• secretKey
• bucket(空间名称)
• domain(空间绑定的 CDN 域名)

✅ 完整 Express + formidable + 七牛 上传示例

const express = require('express');
const { IncomingForm } = require('formidable');
const fs = require('fs');
const path = require('path');
const qiniu = require('qiniu');const app = express();// 七牛配置
const accessKey = '你的AccessKey';
const secretKey = '你的SecretKey';
const bucket = '你的Bucket';
const domain = 'http://你的空间域名'; // 绑定的 CDN 域名(带 http)// 上传凭证
const mac = new qiniu.auth.digest.Mac(accessKey, secretKey);
const putPolicy = new qiniu.rs.PutPolicy({ scope: bucket });
const uploadToken = putPolicy.uploadToken(mac);// 七牛配置对象
const config = new qiniu.conf.Config();
// 根据你空间的 Zone 设置,如华东为 Zone_z0
config.zone = qiniu.zone.Zone_z0;// 表单上传接口
app.post('/upload', (req, res) => {const form = new IncomingForm({ multiples: false, keepExtensions: true });form.parse(req, (err, fields, files) => {if (err) return res.status(500).json({ error: '上传失败' });const file = files.images || files.file;const localFilePath = file.filepath || file.path; // 不同版本字段不同const fileExt = path.extname(file.originalFilename || file.name || '');const fileName = `${Date.now()}-${Math.random().toString(36).slice(2)}${fileExt}`;const putExtra = new qiniu.form_up.PutExtra();const formUploader = new qiniu.form_up.FormUploader(config);formUploader.putFile(uploadToken, fileName, localFilePath, putExtra, (respErr, respBody, respInfo) => {// 删除本地临时文件(可选)fs.unlink(localFilePath, () => {});if (respErr) return res.status(500).json({ error: '七牛上传失败' });if (respInfo.statusCode === 200) {res.json({success: true,url: `${domain}/${respBody.key}` // 返回七牛图片地址});} else {res.status(respInfo.statusCode).json({ error: respBody });}});});
});app.listen(3000, () => {console.log('Server running on http://localhost:3000');
});

✅ 七牛空间设置建议

设置项建议
空间类型公有(或使用私有 + 临时下载链接)
绑定 CDN 域名用于外链图片地址
设置防盗链生产环境建议设置 Referer 防盗链

✅ 最终效果

上传成功后,前端将得到形如:

{
“success”: true,
“url”: “http://cdn.yourdomain.com/1700000000-abc123.jpg”
}

前端就可以将这个 URL 显示为图片头像或多图预览。

腾讯云

如果你要把上传的图片保存到 腾讯云对象存储(COS),可以使用官方的 COS Node.js SDK。

✅ 安装依赖

npm install formidable cos-nodejs-sdk-v5

✅ 腾讯云 COS 准备

你需要准备以下内容:

参数来源(腾讯云后台)
SecretId访问管理 - API 密钥管理
SecretKey同上
Bucket例如:example-1250000000
Region如 ap-nanjing

✅ 后端上传图片到腾讯云 COS

const express = require('express');
const { IncomingForm } = require('formidable');
const COS = require('cos-nodejs-sdk-v5');
const fs = require('fs');
const path = require('path');const app = express();// 腾讯云配置
const cos = new COS({SecretId: '你的SecretId',SecretKey: '你的SecretKey',
});// 上传接口
app.post('/upload', (req, res) => {const form = new IncomingForm({ keepExtensions: true });form.parse(req, (err, fields, files) => {if (err) return res.status(500).json({ error: '上传失败' });const file = files.file || files.image || files.images;const filePath = file.filepath || file.path;const fileExt = path.extname(file.originalFilename || file.name || '');const fileName = `uploads/${Date.now()}-${Math.random().toString(36).slice(2)}${fileExt}`;// 上传到 COScos.putObject({Bucket: '你的Bucket-数字ID',Region: 'ap-你的地域',Key: fileName,Body: fs.createReadStream(filePath),ContentLength: fs.statSync(filePath).size,}, (err, data) => {// 删除临时文件fs.unlink(filePath, () => {});if (err) return res.status(500).json({ error: 'COS上传失败', detail: err });const url = `https://${data.Location}`;res.json({ success: true, url });});});
});app.listen(3000, () => {console.log('Server started on http://localhost:3000');
});

✅ 示例返回

{"success": true,"url": "https://example-1250000000.cos.ap-nanjing.myqcloud.com/uploads/1700000000-abc123.jpg"
}

🛡️ 注意事项

  • Bucket 名称必须是:- 格式,例如:mybucket-1250000000
  • 记得绑定好 CNAME 或设置跨域(CORS)策略以支持前端访问
  • 腾讯云对象存储会自动生成公网访问链接(基于默认域名)

阿里云

如果你想将上传的图片保存到 阿里云 OSS(对象存储服务),可以使用官方的 ali-oss Node.js SDK。

✅ 安装依赖

npm install formidable ali-oss

✅ 准备阿里云 OSS 配置

你需要这些信息:

项目获取方式
accessKeyId阿里云账号控制台 → RAM 用户 → 访问密钥管理
accessKeySecret同上
bucketOSS 控制台创建的 Bucket 名称
regionBucket 所在地域,如 oss-cn-hangzhou
cdnDomain推荐绑定自定义 CDN 域名,如 https://img.example.com

✅ 后端 Express 示例

const express = require('express');
const { IncomingForm } = require('formidable');
const OSS = require('ali-oss');
const fs = require('fs');
const path = require('path');const app = express();// 阿里云 OSS 配置
const client = new OSS({region: 'oss-cn-hangzhou',          // 修改为你的地域accessKeyId: '你的AccessKeyId',accessKeySecret: '你的AccessKeySecret',bucket: '你的Bucket名称',
});const cdnDomain = 'https://your-cdn.example.com'; // 可选:绑定的自定义域名app.post('/upload', (req, res) => {const form = new IncomingForm({ keepExtensions: true });form.parse(req, async (err, fields, files) => {if (err) return res.status(500).json({ error: '上传失败' });const file = files.file || files.image || files.images;const filePath = file.filepath || file.path;const ext = path.extname(file.originalFilename || file.name || '');const fileName = `uploads/${Date.now()}-${Math.random().toString(36).slice(2)}${ext}`;try {const result = await client.put(fileName, filePath);fs.unlink(filePath, () => {}); // 删除临时文件res.json({success: true,url: cdnDomain ? `${cdnDomain}/${fileName}` : result.url,});} catch (ossErr) {res.status(500).json({ error: 'OSS上传失败', detail: ossErr });}});
});app.listen(3000, () => {console.log('Server running at http://localhost:3000');
});

✅ 示例上传成功响应

{"success": true,"url": "https://your-cdn.example.com/uploads/1700000000-abc123.jpg"
}

📝 注意事项

  • 推荐在 OSS 控制台开启跨域(CORS) 支持,方便前端访问
  • 上传后图片即刻可通过公网域名访问
  • 如果你使用的是绑定 CDN 的域名,请确保缓存配置正确(或开启自动刷新)

相关文章:

node.js 实战——express图片保存到本地或服务器(七牛云、腾讯云、阿里云)

本地 ✅ 使用formidable 读取表单内容 npm i formidable ✅ 使用mime-types 获取图片后缀 npm install mime-types✅ js 中提交form表单 document.getElementById(uploadForm).addEventListener(submit, function(e){e.preventDefault();const blob preview._blob;if(!blob)…...

CSS3 伪类和使用场景

CSS3 伪类(Pseudo-classes)大全 CSS3 引入了许多新的伪类,以下是完整的 CSS3 伪类分类列表(包括 CSS2 的伪类): 一、结构性伪类(Structural Pseudo-classes) 这些伪类根据元素在文…...

Shadertoy着色器移植到Three.js经验总结

Shadertoy是一个流行的在线平台,用于创建和分享WebGL片段着色器。里面有很多令人惊叹的画面,甚至3D场景。本人也移植了几个ShaderToy上的着色器。本文将详细介绍移植过程中需要注意的关键点。 1. 基本结构差异 想要移植ShaderToy的shader到three.js&am…...

电脑端音乐播放器推荐:提升你的听歌体验!

在快节奏的职场环境中,许多上班族都喜欢用音乐为工作时光增添色彩。今天要分享的这款音乐工具,或许能为你的办公时光带来意想不到的惊喜。 一、软件介绍-澎湃 澎湃音乐看似是个普通的播放器,实则藏着强大的资源整合能力。左侧功能栏清晰陈列着…...

VIC-2D 7.0 为平面样件机械试验提供全视野位移及应变数据软件

The VIC-2D系统是一个完全集成的解决方案,它基于优化的相关算法为平面试样的力学测试提供非接触、全场的二维位移和应变数据,可测量关注区域内的每个像素子集的面内位移,并通过多种张量选项计算全场应变。The VIC-2D 系统可测量超过 2000%变形…...

一周学完计算机网络之三:1、数据链路层概述

简单的概述 数据链路层是计算机网络体系结构中的第二层,它在物理层提供的基本服务基础上,负责将数据从一个节点可靠地传输到相邻节点。可以将其想象成一个负责在两个相邻的网络设备之间进行数据 “搬运” 和 “整理” 的 “快递中转站”。 几个重要概念…...

网卡网孔速率的协商是如何进行的?

网卡与交换机等网络设备之间的速率协商主要通过**自动协商(Auto-Negotiation)**机制实现,其核心是物理层(PHY)芯片之间的信息交互。以下是协商过程的详细解析: 一、自动协商的核心流程 1. 发送配置帧&am…...

单片机-STM32部分:13-1、蜂鸣器

飞书文档https://x509p6c8to.feishu.cn/wiki/V8rpwIlYIiEuXLkUljTcXWiKnSc 一、应用场景 大部分的电子产品、家电(风扇、空调、电水壶)都会有蜂鸣器,用于提示设备的工作状态 二、原理 蜂鸣器是一种将电信号转换为声音信号的器件&#xff0…...

动态IP技术赋能业务创新:解锁企业数字化转型新维度

在数字经济高速发展的今天,IP地址已不再是简单的网络标识符,而是演变为支撑企业数字化转型的核心基础设施之一。动态IP技术凭借其灵活、高效、安全的特性,正在重塑传统业务模式,催生出诸多创新应用场景。本文将深入剖析动态IP的技…...

使用Python删除PDF中多余或空白的页面

目录 为什么需要删除 PDF 中的多余或空白页面? 所需工具 环境准备 如何使用Python删除PDF中的多余页面 实现思路 详细实现步骤 实现代码 如何使用Python检测并删除PDF中的空白页 实现思路 详细实现步骤 实现代码 在处理 PDF 文件时,常常会遇到…...

英语复习笔记 1

前言 我们知道英语最重要就是单词和阅读理解,因为时间安排和自己懒惰的原因,英语复习实际上进行得非常缓慢。实际上英语复习得比较少,但是我想考一个高分,这样下去肯定是废掉了。所以从今天开始我要好好复习英语。之前有个大佬说…...

UART16550 IP core笔记二

XIN时钟 表示use external clk for baud rate选型&#xff0c;IP核会出现Xin时钟引脚 XIN输入被外部驱动&#xff0c;也就是外部时钟源&#xff0c;那么外部时钟必须要满足特定的要求&#xff0c;就是XIN 的range范围是xin<S_AXI_CLK/2,如果不满足这个条件&#xff0c;那么A…...

TDengine 在金融领域的应用

简介 金融行业正处于数据处理能力革新的关键时期。随着市场数据量的爆炸式增长和复杂性的日益加深&#xff0c;金融机构面临着寻找能够高效处理大规模、高频次以及多样化时序数据的大数据处理系统的迫切需求。这一选择将成为金融机构提高数据处理效率、优化交易响应时间、提高…...

OSCP - Hack The Box - Sau

主要知识点 CVE-2023-27163漏洞利用systemd提权 具体步骤 执行nmap扫描&#xff0c;可以先看一下55555端口 Nmap scan report for 10.10.11.224 Host is up (0.58s latency). Not shown: 65531 closed tcp ports (reset) PORT STATE SERVICE VERSION 22/tcp o…...

大模型的Lora如何训练?

大模型LoRA(Low-Rank Adaptation)训练是一种参数高效的微调方法,通过冻结预训练模型权重并引入低秩矩阵实现轻量化调整。以下是涵盖原理、数据准备、工具、参数设置及优化的全流程指南: 一、LoRA的核心原理 低秩矩阵分解 在原始权重矩阵$ W 旁添加两个低秩矩阵 旁添加两个…...

分词器工作流程和Ik分词器详解

切分词语&#xff0c;normalization 给你一段句子&#xff0c;然后将这段句子拆分成一个一个的单个的单词&#xff0c;同时对每个单词进行 normalization&#xff08;时态转换&#xff0c;单复数转换&#xff09;&#xff0c;分词器 recall&#xff0c;召回率&#xff1a;搜索的…...

QT6 源(93)篇三:阅读与注释共用体类 QVariant 及其源代码,本类支持比较运算符 ==、!=。

&#xff08;9&#xff09; 本类支持比较运算符 、! &#xff1a; 可见&#xff0c; QString 类型里可存储多个 unicode 字符&#xff0c;即使只存储一个 unicode 字符也不等于 QChar。 &#xff08;10&#xff09;本源代码来自于头文件 qvariant . h &#xff1a; #ifndef Q…...

Maven私服搭建与登录全攻略

目录 1.背景2.简介3.安装4.启动总结参考文献 1.背景 回顾下maven的构建流程&#xff0c;如果没有私服&#xff0c;我们所需的所有jar包都需要通过maven的中央仓库或者第三方的maven仓库下载到本地&#xff0c;当一个公司或者一个团队所有人都重复的从maven仓库下载jar包&#…...

力扣210(拓扑排序)

210. 课程表 II - 力扣&#xff08;LeetCode&#xff09; 这是一道拓扑排序的模板题。简单来说&#xff0c;给出一个有向图&#xff0c;把这个有向图转成线性的排序就叫拓扑排序。如果有向图中有环就没有办法进行拓扑排序了。因此&#xff0c;拓扑排序也是图论中判断有向无环图…...

1.1 文章简介

前因后果链 行业需求 → 技能断层 → 课程设计响应 (高薪岗位要求数学基础) → (符号/公式理解困难) → (聚焦原理与应用) 行业驱动因素 • 前因&#xff1a;机器学习/AI等领域的高薪岗位激增&#xff0c;但数学能力成为主要门槛 • 关键矛盾&#xff1a;算法论文中的数学…...

将本地文件上传到云服务器上

使用 SCP 命令&#xff08;最常用&#xff09; # 基本语法 scp [本地文件路径] [用户名][服务器IP]:[目标路径]# 示例&#xff1a;上传单个文件 scp /path/to/local/file.txt root192.168.1.100:/path/to/remote/# 上传整个目录 scp -r /path/to/local/directory root192.168.…...

C++ asio网络编程(5)简单异步echo服务器

上一篇文章:C asio网络编程(4)异步读写操作及注意事项 文章目录 前言一、Session类1.代码2.代码详解3.实现Session类1.构造函数2.handle_read3.介绍一下boost的封装函数和api4.handle_write 二、Server类1.代码2.代码思路详解 三、客户端四、运行截图与流程图 前言 提示&…...

【机器人】复现 UniGoal 具身导航 | 通用零样本目标导航 CVPR 2025

UniGoal的提出了一个通用的零样本目标导航框架&#xff0c;能够统一处理多种类型的导航任务。 支持 对象类别导航、实例图像目标导航和文本目标导航&#xff0c;而无需针对特定任务进行训练或微调。 本文分享UniGoal复现和模型推理的过程&#xff5e; 查找沙发&#xff0c;模…...

基于大模型预测的吉兰 - 巴雷综合征综合诊疗方案研究报告大纲

目录 一、引言(一)研究背景(二)研究目的与意义二、大模型预测吉兰 - 巴雷综合征的理论基础与技术架构(一)大模型原理概述(二)技术架构设计三、术前预测与手术方案制定(一)术前预测内容(二)手术方案制定依据与策略四、术中监测与麻醉方案调整(一)术中监测指标与数…...

spring中的@PropertySource注解详解

一、核心功能与作用 PropertySource是Spring框架中用于加载外部配置文件的核心注解&#xff0c;主要作用是将属性文件&#xff08;如.properties、.yml&#xff09;的键值对加载到Spring的Environment环境中&#xff0c;实现配置与代码的解耦。其核心价值包括&#xff1a; 外部…...

ChromeDriver 技术生态与应用场景深度解析

ChromeDriver 技术生态与应用场景深度解析 随着 Web 自动化测试、运维和数据采集需求的不断增长&#xff0c;ChromeDriver 及其相关技术栈在各行业中扮演着举足轻重的角色。本文将从技术选型、语言适配、典型场景、技术延伸等维度&#xff0c;结合最新行业趋势与实践经验&…...

【NextPilot日志移植】日志写入流程

&#x1f4dd; 文件后端日志写入流程详解 当后端选择文件时&#xff0c;日志写入过程主要涉及 LogWriter 和 LogWriterFile 类的协作。以下是详细的日志写入过程解释及涉及的代码&#xff1a; 1. LogWriter 类初始化 在 LogWriter 类的构造函数中&#xff0c;如果配置的后端…...

二极管钳位电路——Multisim电路仿真

目录 二极管钳位电路 2.1 二极管正向钳位电路 二极管压降测试 2.1.1 二极管正向钳位电路图 2.1.2 二极管正向钳位工作原理 2.2 二极管负向钳位电路 2.2.1 二极管负向钳位电路图 2.2.2 二极管负向钳位工作原理 二极管正向反向钳位仿真电路实验结果 2.3 二极管顶部钳位…...

Docker编排工具---Compose的概述及使用

目录 一、Compose工具的概述 二、Compose的常用命令 1、列出容器 2、查看访问日志 3、输出绑定的公共端口 4、重新构建服务 5、启动服务 6、停止服务 7、删除已停止服务的容器 8、创建和启动容器 9、在运行的容器中执行命令 10、指定一个服务启动容器的个数 11、其…...

5.10-套接字通信 - C++

套接字通信 1.1 通信效率问题 服务器端 单线程 / 单进程 无法使用&#xff0c;不支持多客户端 多线程 / 多进程 写程序优先考虑多线程&#xff1a;什么时候考虑多进程&#xff1f; 启动了一个可执行程序 A &#xff0c;要在 A 中启动一个可执行程序 B 支持多客户端连接 IO 多…...