# 依赖注入

依赖注入(Dependency Injection) 有时候也被叫做 控制反转(IOC), 简称DI。官方文档 (opens new window)

# 概念理解

先看个例子:我们希望在通知组件(NotificationComponent)中通过消息服务(MessageService)发送一条消息。

如果不使用依赖注入的话,我们的代码大概长这样:

class NotificationComponent {
  msg: MessageService;
  constructor() {
     this.msg = new MessageService();
  }
  sendMsg(msgType: string, info: string) {
     this.msg.send(msgType, info);
  }
}

使用依赖注入时:

class NotificationComponent {
  constructor(msg: MessageService) {}  // Angular 中注入依赖的方式
  sendMsg(msgType: string, info: string) {
    this.msg.send(msgType, info);
  }
}
  • 代码的行数变少了
  • NotificationComponent 与 MessageService 间的耦合性降低了

那么现在,我们再来谈谈依赖注入。

不过在说依赖注入之前,我们需要知道有个概念叫做 控制反转,简单来说它和依赖注入间的区别就是:

  • 依赖注入是一种编程技巧
  • 控制反转是一种设计思想

回顾上面例子,我们在 NotificationComponent 的构造函数中引入了 MessageService 并手动实例化,在第二个例子中,我们并没有对实例化这部分做操作,或者说我们把实例化这部分流程交给了外层框架。这种将 依赖的控制权从代码的内部转移到代码的外部 就叫控制反转。

# 多层级注入器 ModuleInjector 与 ElementInjector

alt

ModuleInjector: 通过 @NgModule() 或者 @Injectable() 配置

ElementInjector: 通过 @Directive() 或 @Component() 中的 providers 属性中配置

# 1.ModuleInjector

这部分比较重要的是:

  • Injectable() 的 providedIn 属性是要高于@NgModule() 的 providers 数组
  • 当使用 Injectable() 的 providedIn 属性时,优化工具可以做摇树优化 tree-shaking,比如 providedIn: 'root'
// xxxx.service.ts @Injectable装饰器来告诉系统这个类(服务)是可注入的
@Injectable({
  providedIn: 'root'
})
export class MessageService {
  
}
//xxx.module.ts
@NgModule({
------
providers:[MessageService ]
})

# 2.ElementInjector

对于 ElementInjector 来说比较重要的是:

  • 如果在 NotificationComponent 中声明了 MessageService ,那么每个 NotificationComponent 中都会有一个自己的 MessageService 实例。

# Angular 中的依赖注入使用方式

  • 掌握 @Optional(),@Self(),@SkipSelf(),@Host() 修饰符的含义
  • 熟悉 ClassProvider,ValueProvider,FactoryProvider 使用方法
  • 掌握预定义的 token 与多提供商的巧妙使用

很多 Angular开发者,可能不知道如下的常用写法其实是种简写。

import { Inject } from '@angular/core';
import { MessageService } from '@service/message.service';
@Component({
  selector: 'app-notification',
  templateUrl: './notification.component.html',
  styleUrls: ['./notification.component.less']
})
export class NotificationComponent implements OnInit {
 
  constructor(private service: MessageService) { }
  //实际上是如下简写
  constructor( @Inject(MessageService) private service: MessageService ) { }
 
}

# 修饰符

可以使用 @Optional(),@Self(),@SkipSelf() 和 @Host() 来修饰 Angular 的解析行为。从 @angular/core 导入它们,并在注入服务时在组件类构造函数中使用它们。

Angular 文档中帮助我们对解析修饰符做了分类:

  • 如果 Angular 找不到想要的东西怎么办:@Optional()
  • 到哪里开始寻找,用@SkipSelf()
  • 到哪里停止寻找,用 @Host() 和 @Self()

我们还是通过几个示例来掌握 Angular 中常用的解析修饰符 @Optional(),@Self(),@SkipSelf(),@Host() ,并且试试组合起来的效果吧。

# 1.@Optional()

这表示该服务是可选的,有时候我们引入的服务是不一定存在的,或者说用户不一定会在提供商中配置注入器。

基本的示例:

@Injectable(
  // 注释这段代码,这样在通知组件中就无法找到 MessageService
  // { providedIn: 'root' }
)
export class MessageService {
  constructor() {}
  send() {
    console.log('msg');
  }
}

Angular 提示我们没有 MessageService 的 provider

alt

添加 @Optional() 修饰符之后

@Component({
  selector: 'app-notification',
  templateUrl: './notification.component.html',
  styleUrls: ['./notification.component.less']
})
export class NotificationComponent implements OnInit {
  constructor(@Optional() private msg: MessageService) {}

  ngOnInit() {
    this.msg.send();
  }
}

alt

我们可以看到报错信息有了变化,Angular 将 msg 置为 null,这就是 @Optional() 的作用:

@Optional() 允许 Angular 将您注入的服务视为可选服务。这样,如果无法在运行时解析它,Angular 只会将服务解析为 null,而不会抛出错误

如果你写的业务很简单,你可能很少会用这个修饰符,但是如果你写的是组件库,你就不可避免的需要这个修饰符。例如在 ng-zorro NzAutocompleteComponent 组件中,就有对 noAnimation 的可选修饰(@Host 修饰符接下来就会提到)。

constructor(
    // ...
    @Host() @Optional() public noAnimation?: NzNoAnimationDirective
) {}

当使用 noAnimation 时都会判断是否存在

<div
  #panel
  class="ant-select-dropdown ant-select-dropdown-placement-bottomLeft"
  [class.ant-select-dropdown-hidden]="!showPanel"
  [nzNoAnimation]="noAnimation?.nzNoAnimation"
  [@.disabled]="noAnimation?.nzNoAnimation"
>

# 2.@Self

使用 @Self 让 Angular 仅查看当前组件或指令的 ElementInjector

@Component({
  selector: 'app-notification',
  templateUrl: './notification.component.html',
  styleUrls: ['./notification.component.less']
})
export class NotificationComponent implements OnInit {
  constructor(@Self() private msg: MessageService) {}

  ngOnInit() {
    this.msg.send();
  }
}

由于指明了 Angular 仅查看 Notification 组件,所以会出现如下的错误信息:

alt

这样在组件内提供服务就不会出错了

@Component({
  selector: 'app-notification',
  templateUrl: './notification.component.html',
  styleUrls: ['./notification.component.less'],
  providers: [
    {
      provide: MessageService,
      useClass: NewMessageService
    }
  ]
})
export class NotificationComponent implements OnInit {
  constructor(@Self() private msg: MessageService) {}

  ngOnInit() {
    this.msg.send();
  }
}

然而一般我们会将 @Self 与 @Optional 一起使用,以保证不会抛出异常,ng-zorro 中也将这类写法作为一种最佳实践。例如在 nz-tab-link 指令中

@Directive({
  selector: 'a[nz-tab-link]',
  exportAs: 'nzTabLink'
})
export class NzTabLinkDirective {
  constructor(
    @Optional() @Self() public routerLink?: RouterLink, 
    @Optional() @Self() public routerLinkWithHref?: RouterLinkWithHref
  ) {}
}

这样添加该指令的组件只会对自身的 routerLink 和 routerLinkWithHref 做捕获。

# 3.@SkipSelf

与 @Self 相反

使用 @SkipSelf(),Angular 在父 ElementInjector 中而不是当前 ElementInjector 中开始搜索服务

我们声明 ParentService 并在 container 组件中提供服务

import { Injectable } from '@angular/core';

@Injectable({
  providedIn: 'root'
})
export class ParentMessageService {
  constructor() {}

  send() {
    console.log('come from parent');
  }
}

@Component({
  selector: 'app-container',
  templateUrl: './container.component.html',
  styleUrls: ['./container.component.less'],
  providers: [
    { 
      provide: MessageService, 
      useClass: ParentMessageService 
    }
  ]
})
export class ContainerComponent implements OnInit {
  constructor() {}

  ngOnInit() {}
}

子组件中,我们已提供了服务,但是注入时使用了 @SkipSelf() 修饰符

@Component({
  selector: 'app-notification',
  templateUrl: './notification.component.html',
  styleUrls: ['./notification.component.less'],
  providers: [
    {
      provide: MessageService,
      useClass: NewMessageService
    }
  ]
})
export class NotificationComponent implements OnInit {
  constructor(@SkipSelf() private msg: MessageService) {}

  ngOnInit() {
    this.msg.send();
  }
}

最终可以看到使用的还是父组件内的服务

alt

# 4.@Host

@Host() 修饰符的效果和 @Self() 修饰符的效果很类似,但是作用域的范围,或者说 host 所明确的范围和 @Self() 并不同,

例如我们使用了 ng-content

<app-container>
  <app-notification></app-notification>
</app-container>

这时服务来源于 ContainerComponent alt

或者使用 directive 时也是直接拿到使用该指令的组件中的服务

<app-container appHostTest> </app-container>

alt

在 ng-zorro 中也是大量使用 @Host() 而非 @Self(),在使用 @Host() 时最好也要与 @Optional() 搭配。

# 提供商

ClassProvider,ValueProvider 与 FactoryProvider 我们先来熟悉下这三个提供商,再谈谈应用。

# 1.替代类提供商 ClassProvider

我们常用的提供商配置

{
  // ...
  providers: [
    MessageService
  ],
  // ...
}

其实是这类提供商的简写形式

{
  // ...
  providers: [
    {
      provide: MessageService,
      useClass: MessageService
    }	
  ]
  // ...
}

而替代类提供商可以让不同的类提供相同的服务,例如有个信息更为丰富的 NewMessageService 就可以很方便的替换原有的 MessageService。

{
  // ...
  providers: [
    {
      provide: MessageService,
      useClass: NewMessageService
    }
  ]
  // ...
}

需要注意的是,如果你使用了下面的写法,会创建两个实例

{
  // ...
  providers: [
    NewMessageService,
    {
	provide: MessageService,
	useClass: NewMessageService
    }
  ]
  // ...
}

# 2.值提供商 ValueProvider

有时候不一定要从类去创建对象,直接提供一个现有的对象也可以解决问题。

const MessageServiceValue = {
  send: () => {
    console.log('send message(value provider)');
  }
};

{
  // ...
  providers: [
    {
      provide: MessageService,
      useValue: MessageServiceValue
    }
  ]
	// ...
}

需要注意的是,不要使用一个 TypeScript 中的 interface 来作为 token,因为 interface 只在设计期才会存在(打包后不会存在 interface)。

# 3.工厂提供商 FactoryProvider

这里的工厂就是指工厂函数,既然是函数方式创建对象,那么就拥有了在运行期动态创建的能力。例如区分是否是生产模式来创建不同的对象。

const MessageServiceFactory = () => {
  if (environment.production) {
    return new MessageService();
  } else {
    return new NewMessageService();
  }
};

{
  // ...
  providers: [
    {
      provide: MessageService,
      useFactory: MessageServiceFactory
    }
  ]
  // ...
}

# 预定义的 token 与多提供商(multi 参数)

这里可以理解为 Angular 应用某些操作的回调, 进一步理解 (opens new window)

  • PLATFORM_INITIALIZER:平台初始化之后调用的回调函数
  • APP_BOOTSTRAP_LISTENER:每个启动组件启动完成后的回调函数
  • APP_INITIALIZER:应用初始化之前的回调函数

多提供商机制可以使用一个令牌初始化多个提供商,方法就是设置 multi 参数为 true 即可。

多提供商其实很多人都会用到,比如在设置 HTTP 拦截器时,除了使用默认拦截器之外,还希望再添加上 JWT 拦截器时,多提供商就可以很好的组织服务提供方式:

const INTERCEPTOR_PROVIDES = [
  { provide: HTTP_INTERCEPTORS, useClass: DefaultInterceptor, multi: true },
  { provide: HTTP_INTERCEPTORS, useClass: JWTInterceptor, multi: true }
];

# ng-zorro 源码和 Angular 官方文档的最佳实践

在 Angular 文档中有专门的一节内容 DI 实战 (opens new window),而本文的这部分内容则是通过知名开源 Angular 项目 ng-zorro 来对官方文档的实战篇做些补充,希望可以对各位有所启发。

# 1.使用 @Optional() 来让依赖是可选的,以及使用 @Host() 限定服务方式

这是官方推荐的 @Optional() + @Host() 的组合方式,

举例:

ng-zorro 中关闭动画的 directive nzNoAnimation 可能会被赋予到很多包含动画效果的组件上,拿自动补全组件 NzAutocompleteComponent 来说,构造函数中就会对 noAnimation 添加 @Host() 与 @Optional() 修饰符

export class NzAutocompleteComponent {
  constructor(
    private changeDetectorRef: ChangeDetectorRef,
    private ngZone: NgZone,
    @Host() @Optional() public noAnimation?: NzNoAnimationDirective
  ) {}
}

在使用 noAnimation 时也会加上 ? 以免为 null

<div
	...
  [nzNoAnimation]="noAnimation?.nzNoAnimation"
	...
  [@.disabled]="noAnimation?.nzNoAnimation"
>
...
</div>

# 2.为你的 Angular 应用提供丰富的配置项

在 ng-zorro 中包含很多的配置项用于定制组件样式及行为,比如 NZ_CONFIG,我们来瞧瞧它是怎么实现和使用的吧

在 config.ts 中,我们找到 NG_CONFIG 的定义:

/**
 * User should provide an object implements this interface 
 * to set global configurations.
 */
export const NZ_CONFIG = new InjectionToken<NzConfig>('nz-config');

并且在 NzConfigService 中注入 NZ_CONFIG

@Injectable({
  providedIn: 'root'
})
export class NzConfigService {
  private configUpdated$ = new Subject<keyof NzConfig>();

  /** Global config holding property. */
  private config: NzConfig;

  constructor(@Optional() @Inject(NZ_CONFIG) defaultConfig?: NzConfig) {
    this.config = defaultConfig || {};
  }

  getConfigForComponent<T extends NzConfigKey>(componentName: T): NzConfig[T] {
    return this.config[componentName];
  }

  getConfigChangeEventForComponent(componentName: NzConfigKey): Observable<void> {
    return this.configUpdated$.pipe(
      filter(n => n === componentName),
      mapTo(undefined)
    );
  }

  set<T extends NzConfigKey>(componentName: T, value: NzConfig[T]): void {
    this.config[componentName] = { ...this.config[componentName], ...value };
    this.configUpdated$.next(componentName);
  }
}

在 alert 组件中注入 NzConfigService

constructor(
  public nzConfigService: NzConfigService,
  private cdr: ChangeDetectorRef) 
{
  // 组件配置发生变化时,触发视图更新
  this.nzConfigService
    .getConfigChangeEventForComponent(NZ_CONFIG_COMPONENT_NAME)
    .pipe(takeUntil(this.destroy$))
    .subscribe(() => {
      this.cdr.markForCheck();
    });
}

最后,用户想要使用时,可以通过这样来做:

import { NgZorroAntdModule, NzConfig, NZ_CONFIG } from 'ng-zorro-antd';

const ngZorroConfig: NzConfig = {
  ...
};

@NgModule({
  ...
  providers: [
    { provide: NZ_CONFIG, useValue: ngZorroConfig }
  ],
	...
})
export class AppModule {}

从而实现全局配置

上面我们用到了 InjectionToken 来生成 token,在 Angular 官方文档中,还为我们介绍了如何使用 InjectionToken 来封装浏览器内置的 API,比如 localStorage

首先将 localStorage 改为可注入的 BROWSER_STORAGE token

const BROWSER_STORAGE = new InjectionToken<Storage>('Browser Storage', {
	providedIn: 'root',
	factory: () => localStorage
})

之后声明 BrowserStorageService ,并注入 BROWSER_STORAGE

@Injectable({
  providedIn: 'root'
})
export class BrowserStorageService {
  constructor(@Inject(BROWSER_STORAGE) public storage: Storage) {}

  get(key: string) {
    this.storage.getItem(key);
  }

  set(key: string, value: string) {
    this.storage.setItem(key, value);
  }

  remove(key: string) {
    this.storage.removeItem(key);
  }

  clear() {
    this.storage.clear();
  }
}

# 3.使用 @Self() 和 @SkipSelf() 来修改提供商的搜索方式

@Self() 修饰符的很多作用都被 @Host() 修饰符所替代了,这里我们说一说 @SkipSelf() 在 ng-zorro 中的妙用

使用 ng-zorro 模态框组件 nz-modal 的同学应该都知道 nz-modal 可以通过调用 NzModalService 的 closeAll() 方法来关闭所有的模态框,那这在 ng-zorro 中是如何做到的呢?

我们首先看到 NzModalService 是调用了 NzModalControlService 的 closeAll()

// Closes all of the currently-open dialogs
closeAll(): void {
  this.modalControl.closeAll();
}

NzModalControlService 会去获取当前所有打开的模态框并依次关闭,这并不是通过某个全局变量来存储的,而是通过查找 injection 树来获取的

// Track singleton openModals array through over the injection tree
get openModals(): NzModalRef[] {
  return this.parentService ? this.parentService.openModals : this.rootOpenModals!;
}

来瞧瞧 NzModalControlService 的构造函数中 parentService 的注入方法:

constructor(@Optional() @SkipSelf() private parentService: NzModalControlService) {}

这里就是 @SkipSelf() 的使用方式,与 @Optional() 修饰符搭配可以查找到 Injection 树上所有的注入实例。