在Express中,中间件是处理HTTP请求的函数。它可以访问请求对象(req)、响应对象(res)和应用程序的请求-响应周期。中间件函数可以执行以下操作:
在Express中,中间件可以通过以下方式定义和使用:
使用app.use()
方法:app.use()
方法用于将中间件函数绑定到应用程序的每个请求。它可以在任何路由之前执行,并且对所有请求都生效。
使用路由特定的中间件:可以将中间件函数作为参数传递给特定的路由方法,如app.get()
、app.post()
等。这些中间件函数只对该路由起作用。
使用路径特定的中间件:可以在中间件函数前指定一个路径作为参数,这样只有请求的路径匹配该路径时,对应的中间件才会被执行。
执行顺序取决于它们在代码中的位置。中间件函数按照它们在代码中的顺序依次执行,直到遇到next()
方法。next()
方法将控制权传递给下一个中间件函数。如果在中间件函数中不调用next()
,请求-响应周期将被终止,不会传递给下一个中间件函数或路由处理程序。
中间件在Express中非常强大,它可以用于许多不同的任务,包括身份验证、日志记录、错误处理等。可以使用内置的中间件函数,也可以编写自定义的中间件函数来满足特定的需求。
// index.js
const express = require('express');
const app = express();
app.get('/', (req, res) => {
res.send('这是首页')
console.log('这是首页')
})
const cb1 = function (req, res, next) {
console.log('中间件验证token')
next()
}
// 没有挂载路径的中间件(全局中间件),通过该路由的每个请求都会执行该中间件
app.use(cb1)
const cb2 = function (req, res) {
res.send('查询数据库,返回数据')
console.log('查询数据库,返回数据')
}
// 路由中间件
app.get('/example', [cb2])
app.get('/user',(req,res)=> {
res.send('user页面')
console.log('user页面')
})
app.listen(3000, () => {
console.log('Example app listening on port 3000!');
});
依次访问 /
, /example
, /user
路径,返回值为:
可见在访问 example 和 user 页面之前都会先执行 cb1 方法。
当然,也可以指定中间件执行的路径,比如 app.use('/user',cb1)
此时只在 user 页面会执行 cb1 方法。
路由级中间件和应用级中间件一样,只是它绑定的对象为 express.Router()。 express.Router() 目的是中间件和路由的分离的实例。可以将其视为简略版 express(),只能执行中间件和路由功能。
// index.js
// 写法一:
const express = require('express');
const app = express();
const router = express.Router();
router.use((req, res, next) => {
console.log('Time: ', Date.now())
next()
})
router.get('/', (req, res) => {
res.send('Home page')
})
router.get('/about', (req, res) => {
res.send('About page')
})
app.use('/', router);
app.listen(3000, () => {
console.log('Example app listening on port 3000!');
});
app.use()
和route.use()
都是用于向应用程序或特定路由添加中间件的方法。
app.use()
: app.use()
方法用于将一个中间件函数绑定到应用程序的每个请求。它在任何路由之前,在请求到达服务器时执行。仅当路由路径未指定时,它将被视为全局中间件。
route.use()
: route.use()
方法用于在特定的路由上添加中间件。在router
上的每个请求上执行。router.get()
方法定义了一个特定路由的路由处理程序。
// 写法二:
const express = require('express');
const app = express();
const fs = require('fs')
const path = require('path')
// 声明中间件函数
function recordMiddleware(req, res, next) {
if (req.query.code === '521') {
next()
} else {
res.send('no coding')
}
}
app.get('/home', recordMiddleware, (req, res) => {
res.send('前台首页')
})
app.get('/admin', recordMiddleware, (req, res) => {
res.send('后台首页')
})
app.listen(3000, () => {
console.log('server is running')
})
// 错误中间件放在各个中间件的最后
app.use((req,res,next)=> {
res.status(404).send('Not Found')
})
express.static 是 Express 唯一内置的中间件。它基于 serve-static,负责在 Express 应用中提托管静态资源。每个应用可有多个静态目录。
// 设置静态资源,便于直接访问
// 比如:http://localhost:3000/css/style.css
app.use(express.static(__dirname + '/public'))
注意:
- index.html 为默认访问的资源,即直接访问 / ,也可以访问到 index.html
- 如果静态资源和路由规则同时匹配,谁先匹配谁就响应
安装所需功能的 node 模块,并在应用中加载,可以在应用级加载,也可以在路由级加载。
下面的例子安装并加载了一个解析 cookie 的中间件: cookie-parser
$ npm install cookie-parser
var express = require('express')
var app = express()
var cookieParser = require('cookie-parser')
// 加载用于解析 cookie 的中间件
app.use(cookieParser())