Nestjs 服务 Provider

NestJs仿小米商城项目实战视频教程(视频+课件+源码) https://www.itying.com/goods-1139.html


Nestjs Providers

Providers 是Nest的一个基本概念。许多基本的Nest类可能被视为 provider -service,repository,factory,helper等等。 他们都可以通过constructor注入依赖关系。 这意味着对象可以彼此创建各种关系,并且“连接”对象实例的功能在很大程度上可以委托给Nest运行时系统。 Provider只是一个用@Injectable()装饰器注释的类。

在前面的章节中,我们已经创建了一个简单的控制器CatsController。控制器应处理HTTP请求并将更复杂的任务委托给 providers。Providers 是纯粹的JavaScript类,在其类声明之前带有@Injectable()装饰器。

?> 由于Nest可以以更多的面向对象方式设计和组织依赖性,因此我们强烈建议遵循 SOLID 原则。

Nestjs 服务

让我们从创建一个简单的CatsService开始。该服务将负责数据存储和检索,由其使用CatsController,因此它被定义为provider是一个很好的选择。因此,我们用这个类来装饰@Injectable()。

cats.service.ts

import { Injectable } from '@nestjs/common';
import { Cat } from './interfaces/cat.interface';

@Injectable()
export class CatsService {
  private readonly cats: Cat[] = [];

  create(cat: Cat) {
    this.cats.push(cat);
  }

  findAll(): Cat[] {
    return this.cats;
  }
}

?> 要使用CLI创建服务类,只需执行$ nest g service cats命令。

我们的CatsService是具有一个属性和两个方法的基本类。唯一的新特点是它使用@Injectable()装饰器。该@Injectable()附加有元数据,因此Nest知道这个类是一个Nestprovider。需要注意的是,上面有一个Cat接口。看起来像这样:

export interface Cat {
  name: string;
  age: number;
  breed: string;
}

现在我们有一个服务类来检索cat,让我们在CatsController里使用它 :

cats.controller.ts

import { Controller, Get, Post, Body } from '@nestjs/common';
import { CreateCatDto } from './dto/create-cat.dto';
import { CatsService } from './cats.service';
import { Cat } from './interfaces/cat.interface';

@Controller('cats')
export class CatsController {
  constructor(private readonly catsService: CatsService) {}

  @Post()
  async create(@Body() createCatDto: CreateCatDto) {
    this.catsService.create(createCatDto);
  }

  @Get()
  async findAll(): Promise<Cat[]> {
    return this.catsService.findAll();
  }
}

CatsService是通过类构造函数注入的。注意这里使用了私有的只读语法。这意味着我们已经在同一位置创建并初始化了catsService成员。

Nestjs 依赖注入

Nest 是建立在强大的设计模式, 通常称为依赖注入。我们建议在官方的 Angular文档中阅读有关此概念的精彩文章。

在Nest中,借助 TypeScript 功能,管理依赖项非常容易,因为它们仅按类型进行解析。在下面的示例中,Nest将catsService通过创建并返回一个实例来解析CatsService(或者,在单例的正常情况下,如果现有实例已在其他地方请求,则返回现有实例)。解析此依赖关系并将其传递给控制器的构造函数(或分配给指定的属性):

constructor(private readonly catsService: CatsService) {}

作用域

Provider通常具有与应用程序生命周期同步的生命周期(“作用域”)。在启动应用程序时,必须解析每个依赖项,因此必须实例化每个提供程序。同样,当应用程序关闭时,每个provider都将被销毁。但是,有一些方法可以该标provider生命周期的请求范围。您可以在此处详细了解这些技术。

定制providers

Nest有一个内置的控制反转("IoC")容器,可以解决providers之间的关系。 此功能是上述依赖注入功能的基础,但要比上面描述的要强大得多。@Injectable()装饰器只是冰山一角, 并不是定义 providers 的唯一方法。相反,您可以使用普通值、类、异步或同步工厂。看看这里找到更多的例子。

可选的providers

有时,您可能需要解决一些依赖项。例如,您的类可能依赖于一个配置对象,但如果没有传递,则应使用默认值。在这种情况下,关联变为可选的,provider 不会因为缺少配置导致错误。

要指示provider是可选的,请在constructor的参数中使用@optional()装饰器。

import { Injectable, Optional, Inject } from '@nestjs/common';

@Injectable()
export class HttpService<T> {
  constructor(
    @Optional() @Inject('HTTP_OPTIONS') private readonly httpClient: T
  ) {}
}

请注意,在上面的示例中,我们使用自定义 provider,这是我们包含HTTP_OPTIONS自定义标记的原因。前面的示例显示了基于构造函数的注入,通过构造函数中的类指示依赖关系。在此处详细了解自定义providers及其关联的token。

基于属性的注入

我们目前使用的技术称为基于构造函数的注入,即通过构造函数方法注入providers。在某些非常特殊的情况下,基于属性的注入可能会有用。例如,如果顶级类依赖于一个或多个 providers,那么通过从构造函数中调用子类中的super()来传递它们就会非常烦人了。因此,为了避免出现这种情况,可以在属性上使用@inject()装饰器。

import { Injectable, Inject } from '@nestjs/common';

@Injectable()
export class HttpService<T> {
  @Inject('HTTP_OPTIONS')
  private readonly httpClient: T;
}

!> 如果您的类没有扩展其他provider,你应该总是使用基于构造函数的注入。

注册 provider

现在我们已经定义了 provider(CatsService),并且已经有了该服务的使用者(CatsController),我们需要在Nest中注册该服务,以便它可以执行注入。 为此,我们可以编辑模块文件(app.module.ts),然后将服务添加到@Module()装饰器的providers数组中。

app.module.ts

import { Module } from '@nestjs/common';
import { CatsController } from './cats/cats.controller';
import { CatsService } from './cats/cats.service';

@Module({
  controllers: [CatsController],
  providers: [CatsService],
})
export class AppModule {}

得益于此,Nest现在将能够解决CatsController类的依赖关系。这就是我们目前的目录结构:

src
├── cats
│    ├──dto
│    │   └──create-cat.dto.ts
│    ├── interfaces
│    │       └──cat.interface.ts
│    ├──cats.service.ts
│    └──cats.controller.ts
├──app.module.ts
└──main.ts