第十八章 Express multer 文件上传

本章将学习Express multer 文件上传 ,因为Nest 的文件上传是基于 Express 的中间件 multer 实现的,所以在学习 Nest 文件上传之前,我们先学习下 multer 包

首先先创建 multer-test 文件夹
执行下面代码 创建package.json

npm init -y

1719641865424.png
接着安装 express 和 multer 还有 cors 包:

npm install express multer cors

1719641931393.png
创建index.js 并修改:

const express = require('express')
const multer = require('multer')
const cors = require('cors');const app = express()
app.use(cors());const upload = multer({ dest: 'uploads/' })app.post('/aaa', upload.single('aaa'), function (req, res, next) {console.log('req.file', req.file);console.log('req.body', req.body);
})app.listen(5200);

app.use 使用中间件 cors 来处理跨域。
用 multer 处理文件上传,指定保存目录为 uploads/。
接着新建index.html:

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><script src="https://unpkg.com/axios@0.24.0/dist/axios.min.js"></script>
</head><body><input id="fileInput" type="file" /><script>const fileInput = document.querySelector('#fileInput');async function formData() {const data = new FormData();data.set('name', '光');data.set('age', 20);data.set('aaa', fileInput.files[0]);const res = await axios.post('http://localhost:5200/aaa', data);console.log(res);}fileInput.onchange = formData;</script>
</body></html>

使用FormData + axios 上传文件,指定内容的传输格式 content-type 为 multipart/form-data。
用 node 把 server 跑起来,并且用 http-server 把静态服务跑起来
1719642173152.png
1719642250687.png
游览器访问 http://192.168.100.222:8080/1719642282216.png
接着点击选择文件 上传文件 看游览器的控制台 可以看到发送了请求 此时aaa 请求的 body 是多个 boundary 分隔的格式
1719645151005.png
分隔符是在 Content-Type 指定的
1719645308564.png
这是 form-data 的传输格式
接着我们在控制端可以看到服务端打印了信息
1719645361020.png
服务端多了 uploads 目录,下面就保存着我们上传的文件:
1719645383595.png

我们再实现多文件上传 修改index.js

const express = require('express')
const multer = require('multer')
const cors = require('cors');const app = express()
app.use(cors());const upload = multer({ dest: 'uploads/' })app.post('/aaa', upload.single('aaa'), function (req, res, next) {console.log('req.file', req.file);console.log('req.body', req.body);
})app.post('/bbb', upload.array('bbb',2), function (req, res, next) {console.log('req.files', req.files);console.log('req.body', req.body);
})app.listen(5200);

再次修改index.html

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><script src="https://unpkg.com/axios@0.24.0/dist/axios.min.js"></script>
</head>
<body><input id="fileInput" type="file" multiple/><script>const fileInput = document.querySelector('#fileInput');async function formData2() {const data = new FormData();data.set('name','光');data.set('age', 20);[...fileInput.files].forEach(item => {data.append('bbb', item)})const res = await axios.post('http://localhost:5200/bbb', data);console.log(res);}fileInput.onchange = formData2;</script>
</body>
</html>

input 标签添加 multiple 属性允许多选。
onchange 的时候取出每个 file,通过 append 方法添加到 bbb 字段
1719647069103.png
可以看到上传的是一个数组,并且 uploads 目录下也多了俩文件
1719647129551.png
接着我们在express 里添加错误中间件,一旦某个中间件出了错,express 就会向后找错误处理中间件来调用,如果没有,那就用默认错误处理中间件,返回 500 响应。

const express = require('express')
const multer = require('multer')
const cors = require('cors');const app = express()
app.use(cors());const upload = multer({ dest: 'uploads/' })app.post('/aaa', upload.single('aaa'), function (req, res, next) {console.log('req.file', req.file);console.log('req.body', req.body);
})app.post('/bbb', upload.array('bbb', 2), function (req, res, next) {console.log('req.files', req.files);console.log('req.body', req.body);
}, function (err, req, res, next) {console.log('err', err);
})app.listen(5200);

接着我们上传超过2个文件:
1719647409711.png
可以看到服务端显示了错误,我们再改造一下即可返回错误信息

const express = require('express')
const multer = require('multer')
const cors = require('cors');const { MulterError } = multer; const app = express()
app.use(cors());const upload = multer({ dest: 'uploads/' })app.post('/aaa', upload.single('aaa'), function (req, res, next) {console.log('req.file', req.file);console.log('req.body', req.body);
})app.post('/bbb', upload.array('bbb', 2), function (req, res, next) {console.log('req.files', req.files);console.log('req.body', req.body);
}, function(err, req, res, next) {if(err instanceof MulterError && err.code === 'LIMIT_UNEXPECTED_FILE') {res.status(400).end('Too many files uploaded');}
})app.listen(5200);

传超过 2 个文件,就会收到服务端的 400 的响应:
1719647746031.png

我们尝试多字段上传

const express = require('express')
const multer = require('multer')
const cors = require('cors');const { MulterError } = multer; const app = express()
app.use(cors());const upload = multer({ dest: 'uploads/' })app.post('/aaa', upload.single('aaa'), function (req, res, next) {console.log('req.file', req.file);console.log('req.body', req.body);
})app.post('/bbb', upload.array('bbb', 2), function (req, res, next) {console.log('req.files', req.files);console.log('req.body', req.body);
}, function(err, req, res, next) {if(err instanceof MulterError && err.code === 'LIMIT_UNEXPECTED_FILE') {res.status(400).end('Too many files uploaded');}
})app.post('/ccc', upload.fields([{ name: 'aaa', maxCount: 3 },{ name: 'bbb', maxCount: 2 }
]), function (req, res, next) {console.log('req.files', req.files);console.log('req.body', req.body);
})app.listen(5200);

通过 fields 方法指定每个字段的名字和最大数量,接收到请求后通过 req.files[‘xxx’] 来取对应的文件信息,其他非文件字段,同样是通过 req.body 来取
修改index.html代码:

<!DOCTYPE html>
<html lang="en"><head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title><script src="https://unpkg.com/axios@0.24.0/dist/axios.min.js"></script>
</head><body><input id="fileInput" type="file" multiple /><script>const fileInput = document.querySelector('#fileInput');async function formData2() {const data = new FormData();data.set('name', '光');data.set('age', 20);[...fileInput.files].forEach(item => {data.append('bbb', item)})const res = await axios.post('http://localhost:5200/bbb', data);console.log(res);}async function formData3() {const data = new FormData();data.set('name', '光');data.set('age', 20);data.append('aaa', fileInput.files[0]);data.append('aaa', fileInput.files[1]);data.append('bbb', fileInput.files[2]);data.append('bbb', fileInput.files[3]);const res = await axios.post('http://localhost:5200/ccc', data);console.log(res);}fileInput.onchange = formData3;</script>
</body></html>

可以看到服务端上传了4个新的文件
1719648315461.png
1719648371187.png

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

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

相关文章

深入浅出 Spring @Async 异步编程的艺术

目录 一、异步编程 二、Async 介绍 2.1 Async 使用 三、Async 原理 一、异步编程 在软件开发中&#xff0c;异步编程是非常关键的&#xff0c;尤其是构建高性能、高响应度的应用时。异步编程的主要优势在于它能够避免阻塞操作&#xff0c;提高程序的效率和用户体验。异步编…

修BUG:程序包javax.servlet.http不存在

貌似昨晚上并没有成功在tomcat上面运行&#xff0c;而是直接运行了网页。 不知道为啥又报错这个。。。 解决方案&#xff1a; https://developer.baidu.com/article/details/2768022 就整了这一步就行了 而且我本地就有这个tomcat就是加进去了。 所以说啊&#xff0c;是不是&a…

eNSP公司管理的对象及策略

拓扑图[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传 ) 实验需求 第一步&#xff1a;根据题目搭建拓扑图 其中交换机的型号为&#xff1a;S5700 防火墙设备为&#xff1a;USG6000V 第二步&#xff1a;启动防火墙设备 首先会让你输入密码&#xff0c;…

SQL MySQL定时器/事件调度器(Event Scheduler)

事件调度器&#xff08;Event Scheduler&#xff09;在MySQL数据库系统中是一个强大的功能组件&#xff0c;它允许用户定义一系列称为“事件”的数据库对象&#xff0c;这些事件在指定的时间或时间间隔自动执行预定义的SQL语句或操作。事件调度器通过维护一个时间计划表来管理这…

小抄 20240709

1 很多人做事&#xff0c;没有目标&#xff0c;没有主见&#xff0c;只是按照别人的指示去做&#xff0c;完全不清楚为什么去做。 你去问他为什么要做&#xff0c;他反而要和你急眼&#xff0c;觉得你在质疑他。 2 想要获得超出预期的成功&#xff0c;不是努力到极致&#x…

npm install报错:淘宝镜像证书过期

npm install报错&#xff1a;淘宝镜像证书过期 近期使用npm淘宝镜像新建项目或依赖时出现报错&#xff1a; npm ERR! request to https://registry.npm.taobao.org/xxx failed, reason: certificate has expired 错误原因&#xff1a; 早在 2021 年&#xff0c;淘宝就发文称…

【MySQL】常见的MySQL日志都有什么用?

MySQL日志的内容非常重要&#xff0c;面试中经常会被问到。同时&#xff0c;掌握日志相关的知识也有利于我们理解MySQL 底层原理&#xff0c;必要时帮助我们排查解决问题。 MySQL中常见的日志类型主要有下面几类(针对的是InnoDB 存储引擎): 错误日志(error log):对 MySQL 的启…

QScrollArea 设置最大的高度值

在 Qt 中&#xff0c;QScrollArea 是一个提供滚动视图的控件&#xff0c;允许用户查看大于当前视口尺寸的内容。如果你想要为 QScrollArea 设置一个最大的高度值&#xff0c;这通常不是直接通过 QScrollArea 的属性来设置的&#xff0c;而是需要调整其内容部件&#xff08;widg…

CentOS 6.5配置国内在线yum源和制作openssh 9.8p1 rpm包 —— 筑梦之路

CentOS 6.5比较古老的版本了&#xff0c;而还是有一些古老的项目仍然在使用。 环境说明 1. 更换国内在线yum源 CentOS 6 在线可用yum源配置——筑梦之路_centos6可用yum源-CSDN博客 cat > CentOS-163.repo << EOF [base] nameCentOS-$releasever - Base - 163.com …

新兴市场游戏产业爆发 传音以技术抢抓机遇 ​

随着年轻人口的增加以及互联网的普及,非洲、中东等新兴市场正迎来游戏产业的大爆发,吸引着全球游戏企业玩家在此开疆辟土。中国出海企业代表传音以新兴市场需求为中心,秉持本地化创新理念不断加强游戏等关键领域技术攻关凭借移动终端设备为全球玩家带来极致游戏体验,收获了消费…

就业平台小程序的设计

管理员账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;学生管理&#xff0c;企业管理&#xff0c;企业类型管理&#xff0c;留言板管理&#xff0c;系统管理 微信端账号功能包括&#xff1a;系统首页&#xff0c;招聘信息&#xff0c;简历&#xff0c;我的…

2024年中欧班列累计开行1万列

新华社武汉7月10日电&#xff08;记者王自宸、樊曦&#xff09;今年第10000列中欧班列10日从武汉吴家山站开出&#xff0c;较去年提前19天破万列&#xff0c;累计发送货物108.3万标箱&#xff0c;同比增长11%&#xff0c;为保障产业链供应链稳定、促进中欧经贸往来注入新动能。…

MapReduce底层原理详解:大案例解析(第32天)

系列文章目录 一、MapReduce概述 二、MapReduce工作机制 三、Map&#xff0c;Shuffle&#xff0c;reduce阶段详解 四、大案例解析 文章目录 系列文章目录前言一、MapReduce概述二、MapReduce工作机制1. 角色与组件2. 作业提交与执行流程1. 作业提交&#xff1a;2. Map阶段&…

MATLAB中c2d函数用法

目录 语法 说明 示例 在MATLAB中&#xff0c;c2d函数用于将连续时间系统&#xff08;Continuous-Time System&#xff09;转换为离散时间系统&#xff08;Discrete-Time System&#xff09;。以下是c2d函数的基本语法、说明以及示例&#xff1a; 语法 sys_d c2d(sys_c, T…

【每天认识一个漏洞】spf邮件伪造漏洞

&#x1f31d;博客主页&#xff1a;泥菩萨 &#x1f496;专栏&#xff1a;Linux探索之旅 | 网络安全的神秘世界 | 专接本 | 每天学会一个渗透测试工具 &#x1f3a3;漏洞危害 允许攻击者伪造发件人身份&#xff0c;从而发送钓鱼邮件或垃圾邮件&#xff0c;获取接收方的信任&am…

Spring Boot开发框架

Spring Boot是一个基于Spring框架的开源项目&#xff0c;旨在简化Spring应用的创建、配置和部署。它通过提供默认配置和一系列开箱即用的功能&#xff0c;帮助开发者快速构建生产级的Spring应用。以下是对Spring Boot的详细介绍&#xff1a; 1. 核心理念 1.1 快速入门 Sprin…

SQL 自定义函数

概念 自定义函数是用户根据自己的业务逻辑或计算需求创建的函数。这些函数可以接收一个或多个输入参数&#xff0c;执行一系列的操作&#xff08;如计算、数据处理、逻辑判断等&#xff09;&#xff0c;并最终返回一个值或结果集。自定义函数可以被多次重用&#xff0c;提高了…

C++:cv::boundingRect()函数解析

cv::boundingRect() 函数是 OpenCV 库中的一个函数&#xff0c;用于计算并返回一个点集的边界矩形。这个函数特别有用&#xff0c;当你已经找到了一些轮廓&#xff08;contours&#xff09;或者任何其他形状的点集&#xff0c;并希望获得一个能够包围这些点的最小矩形时。 函数…

编程什么叫f语言编程软件:深入解析F语言编程软件的概念与特性

编程什么叫f语言编程软件&#xff1a;深入解析F语言编程软件的概念与特性 在编程领域&#xff0c;各种编程语言和软件层出不穷&#xff0c;每种都有其独特的特点和适用场景。其中&#xff0c;F语言编程软件作为一种相对较为特殊的存在&#xff0c;引起了众多编程爱好者的关注。…

[leetcode]partition-list 分隔链表

. - 力扣&#xff08;LeetCode&#xff09; class Solution { public:ListNode* partition(ListNode* head, int x) {ListNode *smlDummy new ListNode(0), *bigDummy new ListNode(0);ListNode *sml smlDummy, *big bigDummy;while (head ! nullptr) {if (head->val &l…