Nest.js 系列——Provider 的多种引入方式

前言

Nest中使用了依赖注入的方式,来管理各个模块之间的依赖关系。在Nest中,提供了Provider来注入服务。但是引入有多种方式,这里来记录下每种方法的使用场景和要求。

通过类来引入

在前面知道,provider一般都是用@Injectable装饰的类

tsx 复制代码
@Injectable()
export class AppService {
  getWater(): string {
    return 'hello water'
  }
}

在模块的providers中,通过类来引入,比如服务这种提供者。

tsx 复制代码
import { Module } from '@nestjs/common'
import { AppController } from './app.controller'
import { AppService } from './app.service'

@Module({
  imports: [],
  controllers: [AppController],
  providers: [AppService]
})
export class AppModule {}

其实上面的提供者是一种简写方式,完整的写法如下:

tsx 复制代码
@Module({
  imports:[],
  controllers: [AppController],
  providers: [{
    provide: AppService,
    useClass: AppService,
  }],
})

通过 token 引入

通过 provide 指定注入的 token,通过 useClass 指定注入的对象的类,Nest 会自动对它做实例化再注入

tsx 复制代码
@Controller()
export class AppController {
  constructor(private readonly appService: AppService) {}

  @Get()
  getWater(): string {
    return this.appService.getWater()
  }
}

这里是使用的构造函数注入,也可以通过属性注入。

tsx 复制代码
@Controller()
export class AppController {
  @Inject()
  private readonly appService: AppService

  @Get()
  getWater(): string {
    return this.appService.getWater()
  }
}

通过@Inject装饰器来注入。指定注入的providertoken就行了,而且这个token还可以是一个字符串。

tsx 复制代码
@Module({
  imports: [],
  controllers: [AppController],
  providers: [
    {
      provide: 'app_service',
      useClass: AppService
    }
  ]
})
export class AppModule {}

那么当token是字符串的时候,那么导入的方式就变成下面这样了。

tsx 复制代码
@Controller()
export class AppController {
  constructor(@Inject('app_service') private readonly appService: AppService) {}
  // 或者
  // @Inject('app_service')
  // private readonly appService: AppService;

  @Get()
  getWater(): string {
    return this.appService.getWater()
  }
}

指定一个值

除了指定一个类,还可以指定一个值,比如字符串,数字,布尔值等。

tsx 复制代码
@Module({
  imports: [],
  controllers: [AppController],
  providers: [
    {
      provide: 'water_token',
      useValue: {
        name: 'water',
        age: 18
      }
    }
  ]
})
export class AppModule {}
tsx 复制代码
@Controller()
export class AppController {
  constructor(@Inject('water_token') private readonly waterToken: { name: string; age: number }) {}
}

也可以通过useFactory来动态生成一个值。

tsx 复制代码
@Module({
  imports: [],
  controllers: [AppController],
  providers: [
    {
      provide: 'water_token',
      useFactory: () => {
        return {
          name: 'water',
          age: 18
        }
      }
    }
  ]
})
export class AppModule {}

使用方式与上面一致。同时使用useFactory还支持参数传递。

tsx 复制代码
{
    provide: 'water_token',
    useFactory: (name: string) => {
      return {
        name: name,
        age: 18,
      }
    },
    inject: ['name']
}

provide还可以通过useExisting来指定别名,比如把name这个provider的别名指定为water_token

tsx 复制代码
@Module({
  imports: [],
  controllers: [AppController],
  providers: [
    {
      provide: 'water_token',
      useExisting: 'name'
    }
  ]
})
export class AppModule {}

总结

通过类来引入,通过token来引入,通过useValue来引入一个值,通过useFactory来引入一个值,通过useExisting来引入别名。

以上就是NestProvider的多种引入方式。

相关推荐
java叶新东老师15 分钟前
goland编写go语言导入自定义包出现: package xxx is not in GOROOT (/xxx/xxx) 的解决方案
开发语言·后端·golang
码事漫谈2 小时前
C++模板元编程从入门到精通
后端
_風箏2 小时前
Java【代码 14】一个用于判断磁盘空间和分区表是否需要清理的工具类
后端
_風箏2 小时前
Java【代码 13】前端动态添加一条记后端使用JDK1.8实现map对象根据key的部分值进行分组(将map对象封装成指定entity对象)
后端
_風箏2 小时前
Java【代码 12】判断一个集合是否包含另一个集合中的一个或多个元素 retainAll() 及其他方法
后端
Java中文社群2 小时前
Coze开源版?别吹了!
人工智能·后端·开源
懂得节能嘛.2 小时前
【SpringAI实战】ChatPDF实现RAG知识库
java·后端·spring
站大爷IP3 小时前
Python爬虫库性能与选型实战指南:从需求到落地的全链路解析
后端
小杰来搬砖3 小时前
在 Java 的 MyBatis 框架中,# 和 $ 的区别
后端
wenb1n3 小时前
【安全漏洞】隐藏在HTTP请求中的“隐形杀手”:Host头攻击漏洞深度剖析
java·后端