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

Express中间件和路由及响应方法

1.中间件分类

应用程序级别中间件

        通过 app.use() 或 app.METHOD()(如 app.get)绑定的中间件,作用于整个应用程序。例如

记录请求日志、解析请求体等全局功能。例如:

app.use((req, res, next) => {console.log('Request URL:', req.url);next();
});
路由级别中间件

        绑定到 router.use() 或 router.METHOD() 的中间件,仅对特定路由生效。常用于分组路由的

共享逻辑(如权限校验)。例如:

const router = express.Router();
router.use((req, res, next) => {if (req.user.isAdmin) next();else res.status(403).send('Forbidden');
});
错误处理中间件

        接收四个参数 (err, req, res, next),用于捕获并处理同步/异步错误。需定义在中间件链

末尾。例如:

app.use((err, req, res, next) => {console.error(err.stack);res.status(500).send('Server Error');
});
内置中间件

Express 提供的开箱即用功能,如 express.json() 解析 JSON 请求体:

app.use(express.json());
第三方中间件

通过 npm 安装的中间件,例如 cors 处理跨域、helmet 增强安全性:

app.use(require('cors')());

2.中间件的使用

app.js文件中:

const express = require("express");const app = express();
const routerIndex=require("./router/index")
const videoRouter=require("./router/vedio")
const PORT = process.env.PORT || 3000;// 设置日志中间件 中间件代码的位置很有讲究 必须是所有路由的前面才行
app.use((req, res, next) => {console.log(`${req.method} ${req.url},${Date.now()}`);// 继续处理下一个中间件或路由处理函数next();
},function(req, res, next) { console.log('输出了日志,然后就可以执行这里面的代码,可以写多个这样的函数,然后使用next()执行下一个函数');next();
});
// 当有四个参数时,会默认这是个错误处理中间件
app.use((err,req,res,next)=>{console.log(err,'错误信息');res.status(500).send("服务器错误")
})
// 挂载路由  所有的路由前面都要加 /api,相当于vue的反向代理
app.use('/api',routerIndex);
app.use('/vedio',videoRouter)
// 挂载统一处理服务端错误中间件
// app.use(errorHandler());// all 方法,可以处理 /xx 路径传过来的所有请求方法都可以触发,不管时get、post、put请求都会触发它
app.all('/xx',(req,res)=>{res.send('xx')
})app.listen(PORT, () => {console.log(`Server is running at http://localhost:${PORT}`);
});

在app.js文件同级的router文件夹下的index.js:

const express = require('express')const router = express.Router()router.get('/',(req,res)=>{console.log(req.method);res.send('/index')
})
router.get('/users',(req,res)=>{console.log(req.method);res.send('/users')
})module.exports = router

在app.js文件同级的router文件夹下的vedio.js:

const express =require('express')
const router=express.Router()
// 这里的路由不是前端的跳转页面路由,而是后端的接口路由路径
router.get('/list',(req,res)=>{console.log('list');res.send('list')
})
module.exports=router

3.路由

// 可以匹配正则,/us+er表示匹配多个s叠加的路径,比如/usser、/ussser
app.get('/us+er',(req,res)=>{res.send(`${req.method} ${req.url}`)
})

获取路由参数:

router.get('/list/:id',(req,res)=>{// 获取路由参数console.log(req.params.id);res.send('list')
})

还可以链式调用:

router.get('/list/:id/are/:age',(req,res)=>{// 获取路由参数console.log(req.params.id);  // {id:2,age:3}res.send('list')
})
.post('/vedio',(req,res)=>{res.send('list')
})

使用res.render渲染:

// app.js文件
const express = require("express");
const app = express();
const path = require('path');
// 设置模板引擎(以 EJS 为例)
app.set('view engine', 'ejs');
// 设置视图文件目录
app.set('views', path.join(__dirname, 'views'));
const routerIndex=require("./router/index")
// 所有的路由前面都要加 /api,相当于vue的反向代理
app.use('/api',routerIndex);// router/index.js文件
const express = require('express')
const router = express.Router()
router.get('/',(req,res)=>{console.log(req.method);// res.render 是服务器端渲染,页面跳转到views/test/index.html页面中 res.render('test/index', { name: 'Express 路由1' });// res.send('/index')
})
module.exports = router// views/test/index.ejs文件
<!DOCTYPE html>
<html lang="zh-CN"><head><metaname="viewport"content="width=device-width, initial-scale=1, maximum-scale=1, user-scalable=no"/><meta http-equiv="Content-Type" content="text/html; charset=UTF-8" /><title><%= name %></title></head><body><h1>Hello, <%= name %></h1></body><script>function corsGetData() {alert("dsafdsafdsafsda");}</script>
</html>// 安装ejs
npm install ejs

效果:(请求 http://127.0.0.1:3000/api/ 直接渲染页面)

其他的响应方法:

// 这里的路由不是前端的跳转页面路由,而是后端的接口路由路径
router.get('/list/:id',(req,res)=>{// 获取路由参数console.log(req.params.id);// 下载文件  提示客户端下载文件res.download('./files/report.pdf')// 结束响应过程  不发送任何数据,直接结束响应// 通常用于简单响应或与 res.write() 配合使用res.end()// 发送 JSON 格式的响应 自动设置 Content-Type 为 application/jsonres.json({ message: 'Hello World' })// 重定向到指定的 URL  默认发送 302 状态码res.redirect('/login')// 渲染模板引擎视图res.render('index.ejs', { title: '首页' })// res.render 是服务器端渲染,页面跳转到views/test/index.html页面中 res.render('test/index', { name: 'Express 路由1' });// 发送文件作为响应res.sendFile('/views/index.html')// 设置响应状态码并发送其字符串表示作为响应体res.sendStatus(404) // 会发送 404 状态码和 "Not Found" 字符串res.send('list')
})

4. 设置跨域和记录日志

// 跨域
npm i cors
// 请求日志中间件
npm i morgan

在app.js文件中:

const cors =require('cors')
const morgan =require('morgan')
app.use(cors())
app.use(morgan('dev'))// 只记录开发时的日志
http://www.lryc.cn/news/614946.html

相关文章:

  • golang的二维数组
  • vulnhub-Beelzebub靶场通关攻略
  • Nginx 功能扩展与二次开发实践
  • 目标检测数据集 - 无人机检测数据集下载「包含COCO、YOLO两种格式」
  • 1.JavaScript 介绍
  • 130Kw双向储能PCS电源及关键技术分析
  • 彻底解决vscode中fnm调用失败的问题
  • 嵌入式 Linux Mender OTA 实战全指南
  • Microsoft 365中的Message Encryption (Basic)功能深度解析
  • 【JVM】深入解析Java虚拟机
  • Vitalik谈以太坊:ETH财库储备策略“有益且有价值”
  • Jmeter性能测试之安装及启动Jmeter
  • 检索增强生成:RAG(Retrieval Augmented Generation)
  • 如何在linux(CentOS7)上面安装 jenkins?
  • 【Vapor Mode】Vue 从“运行时“优化转向“编译时“优化的范式跃迁
  • 浏览器自动播放策略
  • OpenAI发布的GPT-5 更新了哪些内容,它的核心能力有哪些?AI编码能力这么强,前端程序员何去何从?
  • FreeRTOS学习笔记:任务通知和软件定时器
  • SpringBoot学习日记 Day6:解锁微服务与高效任务处理
  • 39.【.NET8 实战--孢子记账--从单体到微服务--转向微服务】--扩展功能--调整发布脚本
  • 24SpringCloud黑马商城微服务整合Seata重启服务报错的解决办法
  • Web3: DeFi借贷的安全基石, 了解喂价与清算机制的原理与重要性
  • 递归---记忆化搜索
  • 八、Linux Shell 脚本:变量与字符串
  • ESP32之wifi_HTTP
  • 商品、股指及ETF期权五档盘口Tick级与分钟级历史行情数据多维解析
  • 网盘短剧资源转存项目源码 支持垮克 带后台 附教程
  • 深入解析 Apache APISIX 在微服务网关中的性能优化实践指南
  • LeetCode 面试经典 150_数组/字符串_分发糖果(15_135_C++_困难)(贪心算法)
  • Swift 实战:秒算两个数组的交集(LeetCode 349)