NestJS 的 中间件 学习

基本概念

中间件是在路由处理程序之前调用的函数。中间件函数可以访问请求响应对象。在程序中我们可以让多个中间件串起来一起使用,当多个中间件一起使用时我们可以使用next()调用下一个中间件。

中间件主要是可以实现如下功能:

  • 执行任何代码
  • 更改请求和响应对象
  • 结束请求-响应周期
  • 调用堆栈中的下一个中间件函数
  • 如果当前的中间件函数没有结束请求-响应周期,则必须调用 next()将控制权传递给下一个中间件函数。否则,请求将被挂起

在 NestJS 中可以在函数或者带有@Injectable()装饰器装饰的类来定义中间件,并且使用类时我们必须实现NestMiddleware接口,函数不需要做特殊处理。

依赖注入

在 NestJs 中中间件可以使用依赖注入的方式进行注入使用,就像控制器和提供者一样进行注入。

需要使用的时候,我们可以通过constructor进行实例化,然后进行使用。

在模块中使用中间件

@Module装饰器装饰的类中,我们是没有地方进行注入的,我们只能在类中使用configure()来声明使用中间件,并且类还必须集成NestModule接口。

接下来我们声明一个日志中间件,主要是简单的把请求对象和响应对象打印一下,具体代码如下:

typescript 复制代码
// 日志中间件
import { Injectable, NestMiddleware } from "@nestjs/common";
import { Request, Response, NextFunction } from "express";

@Injectable()
export class LoggerMiddleware implements NestMiddleware {
  use(req: Request, res: Response, next: NextFunction) {
    console.log("-------");
    console.log(req);
    console.log(res);
    console.log("-------");
    next();
  }
}
typescript 复制代码
// App模块
@Module({
  imports: [CommodityModule, AccountModule],
})
export class AppModule implements NestModule {
  configure(consumer: MiddlewareConsumer) {
    consumer.apply(LoggerMiddleware).forRoutes("commodity");
  }
}

在上面这里例子中,我们的中间件主要是针对带有commodity前缀的所有请求路径进行处理,并且注入的中间件为LoggerMiddleware

我们还可以在forRoutes传入参数对象,从而过滤出符合的请求进入到中间件中进行处理。例如我们限定监听请求路径是commodity并且是POST的请求:

typescript 复制代码
configure(consumer: MiddlewareConsumer) {
  consumer
    .apply(LoggerMiddleware)
    .forRoutes({ path: 'commodity', method: RequestMethod.POST });
}

forRoutes 的路由通配符

在上述的例子中,我们在forRoutes中传入路由参数后筛选出匹配的路由地址,其中path是匹配的路由地址,我们可以使用通配符来定义匹配的路由,而通配符设置规则已经在NestJS 的 Controller 学习文章中介绍过。

中间件消费者(MiddlewareConsumer 类)

MiddlewareConsumer是一个辅助类,它提供了几种内置方法来管理中间件。

forRoutes()方法传入的参数可以是字符串、多个字符串、RouterInfo 对象和控制器类,建议使用控制器类来作为forRoutes的参数,因为使用控制器类后,我们不需要过多的来维护监听的数据,这样维护效率会比较高并且清晰。

排除路线

有些时候我们希望排除某些路由应用中间件的时候,我们可以使用exclude()方法来排除某些路由。该方法传入的参数可以是字符串、多个字符串和 RouterInfo。具体的例子如下:

typescript 复制代码
configure(consumer: MiddlewareConsumer) {
  consumer
    .apply(LoggerMiddleware)
    .exclude('account/(.*)')
    .forRoutes(CommodityController, AccountController);
}

函数中间件

当我们的中间件不需要任何成员、方法和依赖项的时候,可以使用函数中间件来替代类中间件。函数中间件的例子我们还是采用日志中间件为例,具体的例子如下:

typescript 复制代码
export function MyLogger(req: Request, res: Response, next: NextFunction) {
  console.log("-------");
  console.log(req);
  console.log(res);
  console.log("-------");
  next();
}

全局中间件

全局中间件可以在main.ts文件中进行定义使用,具体如下:

typescript 复制代码
const app = await NestFactory.create(AppModule);
app.use(logger);
await app.listen(3000);
相关推荐
知识分享小能手24 分钟前
React学习教程,从入门到精通, React 属性(Props)语法知识点与案例详解(14)
前端·javascript·vue.js·学习·react.js·vue·react
茯苓gao3 小时前
STM32G4 速度环开环,电流环闭环 IF模式建模
笔记·stm32·单片机·嵌入式硬件·学习
是誰萆微了承諾3 小时前
【golang学习笔记 gin 】1.2 redis 的使用
笔记·学习·golang
DKPT4 小时前
Java内存区域与内存溢出
java·开发语言·jvm·笔记·学习
aaaweiaaaaaa4 小时前
HTML和CSS学习
前端·css·学习·html
看海天一色听风起雨落5 小时前
Python学习之装饰器
开发语言·python·学习
speop6 小时前
llm的一点学习笔记
笔记·学习
非凡ghost6 小时前
FxSound:提升音频体验,让音乐更动听
前端·学习·音视频·生活·软件需求
ue星空7 小时前
月2期学习笔记
学习·游戏·ue5
萧邀人7 小时前
第二课、熟悉Cocos Creator 编辑器界面
学习