中间件是在接收到请求和发送响应中间的一系列操作。事实上,express是一个路由和中间件的web框架,Express 应用程序基本上是一系列中间件函数的调用。
中间件函数可以执行以下任务:
- 执行任何代码。
- 对请求和响应对象进行更改。
- 结束请求/响应循环。
- 调用堆栈中的下一个中间件函数。
中间件也分为应用层中间件、路由中间件、内置中间件、错误处理中间件和第三方中间件。
一、应用层中间件
应用级别中间件绑定到app对象上,使用app.use和app.METHOD()(METHOD:需要处理http请求的方法,例如GET、PUT、POST)
var express = require("express");
var app = express();
//匹配路由之前的操作
app.use(function(req,res,next()){
console.log("访问之前");
});
app.get("/",function(req,res){
res.send("主页");
});
app.listen(8080);
这时我们发送http://localhost:8080/请求,地址一直在加载,但命令行里显示了“访问之前”,因为我们并没有调用res对象的任何方法,如果想要得到字符串“主页”的响应内容,下面代码会如你所愿。
var express=require("express");
var app=express();
//匹配路由之前的操作
app.use(function(req,res,next){
console.log("访问之前");
//此处调用了next()方法,则程序才会往下执行app.get处理请求并响应字符串“主页”
next();
});
app.get("/",function(req,res){
res.send("主页");
});
app.listen(8080);
下面代码是简化版的写法
var express=require("express");
var app=express();
app.use(function(req,res,next){
console.log("访问之前");
next();
},function(req,res){
res.send("主页");
});
app.listen(8080);
因此,在进行路由匹配之前执行某些程序处理,再继续向下执行匹配的路由函数,那么应用层中间件无疑是好的选择。
二、路由中间件
路由级中间件和应用级中间件类似,只不过他需要绑定express.Router();
var router = express.Router()
在匹配路由时,我们使用 router.use() 或 router.METHOD()(METHOD:需要处理http请求的方法,例如GET、PUT、POST),结合多次callback可用于用户登录及用户状态检测。
var express = require("express");
var app = express();
var router=express.Router();
router.use("/",function(req,res,next){
console.log("匹配前");
next();
});
router.use("/user",function(req,res,next){
console.log("匹配地址:",req.originalUrl);
next();
},function(req,res){
res.send("用户登录");
});
app.use("/",router);
app.listen(8080);
总之在检测用户登录状态和权限判断、引导用户应该访问哪个页面时,路由中间件是不错的选择。
三、错误处理中间件
顾名思义,它是指当我们匹配不到路由时所执行的操作。错误处理中间件和其他中间件基本一样,只不过其需要开发者提供4个自变量参数。
app.use((err, req, res, next) => {
res.sendStatus(err.httpStatusCode).json(err);
});
一般情况下,我们把错误处理放在最下面,这样我们即可对错误进行集中处理。
var express = require("express");
var app = express();
app.get("/login",function(req,res,next){
if(req.query.username == 'david' && req.query.password == 123456){
res.send("主页");
}else{
const err=new Error("用户名或密码错误");
next(err);//直接调用app.use(function(err, req, res, next){})
}
});
//如果前面没有匹配的请求,则会执行此函数
app.use(function(req, res, next) {
next(404);
});
app.use(function(err, req, res, next) {
if(err === 404){
res.status(404).send("未找到指定页面");
}else{
res.status(500).send(err.message);
}
});
四、内置中间件
从版本4.x开始,Express不再依赖Content,也就是说Express以前的内置中间件作为单独模块,express.static是Express的唯一内置中间件。
express.static(root, [options]);
通过express.static我们可以指定要加载的静态资源,在我们访问:http://localhost:3000/public
app.use(express.static(path.join(__dirname, 'public')));
五、第三方中间件
如body-parser,采用引入外部模块的方式来获得更多的应用操作。如cookie和session。
var express = require('express');
var app = express();
var cookieParser = require('cookie-parser');
app.use(cookieParser());
在实际项目中,中间件都是必不可少的,所以熟悉使用各种中间件可以加快项目的开发效率。