gpt4 book ai didi

【NestJS系列】核心概念:Module模块

转载 作者:我是一只小鸟 更新时间:2023-08-24 14:31:24 36 4
gpt4 key购买 nike

                        
                          theme: fancy
highlight: atelier-dune-dark

                        
                      

前言

模块指的是使用 @Module 装饰器修饰的类,每个应用程序至少有一个模块,即 根模块 。根模块是 Nest 用于构建应用程序的起点,理论上 Nest 程序可能只有根模块,但在大多数情况下是存在多个模块的,每个模块各自封装一组相关的功能.

@Module装饰器

@Module() 装饰器可以传入一个对象,属性值如下:

providers 将由 Nest 注入器实例化的提供程序,并且至少可以在该模块中共享
controllers 该模块中定义的必须实例化的控制器集
imports 导入模块的列表,导出该模块所需的提供程序
exports 该子集 providers 由该模块提供,并且应该在导入该模块的其他模块中可用
                        
                          @Module({
  imports: [NanjiuModule, UserModule],
  controllers: [AppController],
  providers: [AppService],
})
export class AppModule {}

                        
                      

模块共享

如果你想把当前模块的 service 暴露给其它模块使用,则可以使用 exports 到处该服务 。

比如我使用 nest g resource info 新建了一个info类,并且使用 export 导出该服务 。

                        
                          // info.module.ts
import { Module } from '@nestjs/common';
import { InfoService } from './info.service';
import { InfoController } from './info.controller';

@Module({
  controllers: [InfoController],
  providers: [InfoService], // 提供者
  exports: [InfoService] // 导出 InfoService 供其他模块使用
})
export class InfoModule {}


                        
                      

然后我在 user 模块中使用 imports 导入该模块 。

                        
                          // user.module.ts
import { Module } from '@nestjs/common';
import { UserService } from './user.service';
import { UserController } from './user.controller';
import { InfoModule } from 'src/info/info.module';

@Module({
  imports: [InfoModule], // 导入 InfoModule
  controllers: [UserController],
  providers: [UserService]
})
export class UserModule {}


                        
                      

最后在 controller 中依赖注入并使用 。

                        
                          // user.controller.ts
import { InfoService } from 'src/info/info.service';

@Controller('user')
export class UserController {
  constructor(
    private readonly userService: UserService,
    private readonly infoService: InfoService, // 注入 InfoService
    ) {}

  @Post()
  create(@Body() createUserDto: CreateUserDto) {
    return this.infoService.findAll() // 调用 InfoService 的 findAll 方法

    // return this.userService.create(createUserDto);
  }
  //...
}

                        
                      

这样就完成模块共享了,可以看到我们在 user 模块中可以调用 info 的服务 。

模块再导出

可以把一些常用的,公共的模块,全部先import进一个CommonModule,然后再把它们从exprots全部导出,以后如果有那个模块想要使用其中某个模块的Service,只需要将这个CommonModule导入即可,不用再导入所有的依赖模块 。

                        
                          // common.module.ts
@Module({
  imports: [Module1, Module2, Module3, Module4],
  exports: [Module1, Module2, Module3, Module4],
})
export class CommonModule {}

                        
                      

依赖注入

模块类也可以注入 provider 服务 。

                        
                          
@Module({
  controllers: [UserController],
  providers: [UserService],
})
export class UserModule {
  constructor(private userService: UserService) {}
}

                        
                      

全局模块

通过 @Global() 装饰器声明一个全局模块,只需要在根模块 imports 注册该全局模块,就可以在其他所有模块内使用它导出的 Service 。

比如:将 info 声明为全局模块 。

                        
                          // info.module.ts
@Global() // 全局模块
@Module({
  controllers: [InfoController],
  providers: [InfoService], // 提供者
  exports: [InfoService] // 导出 InfoService 供其他模块使用
})
export class InfoModule {}


                        
                      

然后在 user 模块中无需导入,只需依赖注入就可直接使用(前提是已在根模块导入) 。

                        
                          // user.controller.ts
import { CreateUserDto } from './dto/create-user.dto';
import { InfoService } from 'src/info/info.service';

@Controller('user')
export class UserController {
  constructor(
    private readonly userService: UserService,
    private readonly infoService: InfoService, // 注入 InfoService
    ) {}

  @Post()
  create(@Body() createUserDto: CreateUserDto) {
    return this.infoService.findAll() // 调用 InfoService 的 findAll 方法
  }
}

                        
                      

动态模块

动态模块能够让我们创建可定制的模块,当导入模块并向其传入某些选项参数,这个模块根据这些选项参数来动态的创建不同特性的模块.

创建动态模块

动态模块其实就是给当前 Module 类提供一个 forRoot 方法,该方法返回一个新的 Module ,这个Module的类型是一个DynamicModule,在其他模块需要注册使用时,可以使用 xxxModule.forRoot(args) 来动态的注册不同的Module,以达到提供不同providers的目的.

这里我们创建一个 config 的动态模块 。

                        
                          // config.module.ts
import { Module, DynamicModule, Global } from '@nestjs/common';
import { NanjiuService } from 'src/nanjiu/nanjiu.service';
import { UserService } from 'src/user/user.service';

interface Options {
    name: string
}
@Global()
@Module({
})
export class ConfigModule {
    static forRoot(options: Options): DynamicModule {
        console.log('options', options)
        return {
            module: ConfigModule,
            providers: [
                {provide: 'config', useClass: options.name === 'nanjiu' ? NanjiuService : UserService},
            ],
            exports: [
                {provide: 'config', useClass: options.name === 'nanjiu' ? NanjiuService : UserService}
            ]
        }
    }
}

                        
                      

这个例子很简单,首先需要自己编写一个静态方法,该方法通过接收传递进来的参数判断使用哪一个 service ,并且为了方便,我这里直接使用 @Global() 装饰器将该模块声明称了全局模块 。

传递参数使用

调用静态方法传递参数 。

                        
                          // app.module.ts
@Module({
  imports: [ConfigModule.forRoot({name: 'fe'})],
  controllers: [AppController],
  providers: [AppService],
})
export class AppModule {}

                        
                      

然后在 controller 中使用 。

                        
                          import { Controller, Get, Inject } from '@nestjs/common';
import { AppService } from './app.service';

@Controller()
export class AppController {
  constructor(
    private readonly appService: AppService, 
    @Inject('config') private readonly configService // 注入 ConfigService
    ) {}

  @Get('/hello2')
  get2() {
    return this.configService.getHello() // 调用 ConfigService 的 getHello 方法
  }
}

                        
                      

比如上面 forRoot 传递的参数是 {name: 'nanjiu'} ,所以此时的 ConfigModule 注入的应该是 UserService 。

修改 forRoot 参数 。

                        
                          // app.module.ts
@Module({
  imports: [ConfigModule.forRoot({name: 'nanjiu'})],
  controllers: [AppController],
  providers: [AppService],
})
export class AppModule {}

                        
                      

此时通过 get 方式再访问同样的路由,应该是访问到 NanjiuService 提供的服务了.

以上就是动态模块的简单用法,后续内容我们还会再遇到它~ 。

最后此篇关于【NestJS系列】核心概念:Module模块的文章就讲到这里了,如果你想了解更多关于【NestJS系列】核心概念:Module模块的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。

36 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com