使用依赖注入框架管理多实例服务(以 InversifyJS 为例)

在大型项目的管理中,控制反转的思想是非常重要的。它可以帮助我们解耦代码,提高代码的可维护性。同时避免了不必要的重复实例化,降低内存泄漏的可能性。

而在 JS/TS 技术栈中,我们通常会使用依赖注入框架来帮助我们管理服务。这其中最佳的选择当然是 Angular 这种大而全的大型工程开发框架。而对于使用了其他 UI 框架的项目来说,我们同样可以额外引入一个轻量化的依赖注入框架。而 InversifyJS 就是其中的佼佼者。我们可以通过使用它,来见微知著地了解依赖注入的原理与设计哲学。

但最近在使用 Inversify 进行项目重构时,遇到了一个问题:众所周知依赖注入框架天生适合管理单例服务。它的设计哲学是 `Everything as Service`。但是在某些场景下,单例模式并不能解决一切问题,我们同样需要进行多实例的管理。那么我们该如何解决这个问题呢?

这并不是 Inversify 框架的问题,而其实是一个依赖注入框架下常见的设计疑惑,但是网上对此的解析资料却很少。

我看了很多使用了 InversifyJS 的项目,他们对此的方式就是直接在需要处实例化,不将其注册到容器中。这实际上是没有真正理解到依赖注入框架的内核。这样做的好处是简单,但是有很多弊端。由于我们无法在容器中统一管理这些实例,那么这些服务的生命周期将不受控制,在 dispose 时无法在容器中统一销毁这些实例。与不引入依赖注入框架一样,这样同样会带来内存泄漏的可能性。

那么该如何正确地处理这种情况呢?

## 构造器注入

一个最简便的改造方式是,我们将类的构造函数绑定到容器中。需要的时候从容器中获取类的构造器,再进行实例化。这样我们就可以在容器中统一管理这些实例了。

```ts

// 将 InstanceClass 的构造函数绑定到容器中

container

  .bind<interfaces.Newable<InstanceClass>>("Newable<InstanceClass>")

  .toConstructor<InstanceClass>(InstanceClass);

```

```ts

// 获取构造器

public constructor(

    @inject("Newable<InstanceClass>") InstanceClass: Newable<InstanceClass>,

) {

    this.instance1 = new InstanceClass();

    this.instance2 = new InstanceClass();

}

```

实例会跟随类的生命周期而存在,且该类能纳入容器中进行管理。但是这样做,实际上仍然无法在容器中统一管理这些实例的生命周期。如果我们需要在 dispose 时销毁这些实例,那么我们需要在类中手动实现 dispose 方法,并在 dispose 时手动销毁这些实例。

这样改造的好处是简单,但是很多时候并不是一个最优解,因为我们希望该实例本身能在注入框架的管理下,避免我们去手动的控制与销毁。

## 工厂注入

依赖注入框架天生不太好管理多实例的服务,但是如果利用工厂模式的设计思想,将这些服务的实例化过程封装到工厂中,而这样的工厂类一定是单例的。那么我们就可以通过管理工厂类来管理这些多实例服务的生命周期了。

在需要多实例服务实例化时,我们不直接 import 类进行实例化,而是通过 import 工厂类来获取实例。这样我们就可以在工厂中控制多实例服务的生命周期了。

在 InversifyJS 中,提供了工厂注入的方法:

```ts

// 设置工厂函数

const instanceFactory = () => {

  return context.container.get<InstanceClass>("Instance");

};

// 工厂创建器,这里设置高阶函数的目的是将 context 传递给工厂函数,方便获取容器

const instanceFactoryCreator = (context: interfaces.Context) => {

  return instanceFactory;

};

// 绑定工厂

container

  .bind<interfaces.Factory<InstanceClass>>("Factory<InstanceClass>")

  .toFactory<InstanceClass>(instanceFactoryCreator);

```

```ts

// 获取构造器

public constructor(

    @inject("Factory<InstanceClass>") private instanceFactory: () => InstanceClass,

) {

    this.instance1 = this.instanceFactory();

    this.instance2 = this.instanceFactory();

}

```

这样的实现非常优雅,也是 Inversify 推荐的多实例管理方式。

当然,你也可以通过高阶函数的方式,生成不同的的工厂函数,以实现不同的实例化逻辑。

```ts

// 设置工厂函数

const instanceFactory = (name: string) => () => {

  if (name === "Instance") {

    return context.container.get<InstanceClass>("Instance");

  }

  return context.container.get<DefaultClass>("Default");

};

// 工厂创建器,这里设置高阶函数的目的是将 context 传递给工厂函数,方便获取容器

const instanceFactoryCreator = (context: interfaces.Context) => {

  return instanceFactory;

};

// 绑定工厂

container

  .bind<interfaces.Factory<InstanceClass>>("Factory<InstanceClass>")

  .toFactory<InstanceClass>(instanceFactoryCreator);

```

在大多数情况下,它就是最标准的依赖注入框架下多实例管理方式了,也推荐能使用此方式的类尽量如此改造。

## 带参数实例化的工厂注入

现在重点来了,依赖注入框架完美解决了在类实例化时需要传入的依赖实例,避免了我们需要在类的构造函数中获取或新建依赖实例。那么,对于那些依赖于传入外部上下文变量的类,我们该如何处理呢?

这是我们将已有的项目重构的过程中,经常会遇到的一种情况,这些类的构造函数执行过程依赖于外部上下文变量。

InversifyJS 的工厂注入在这中情形下的推荐实现方式比较奇怪,是在获取实例后为实例进行属性注入。我大致转写一下主要实现:

```ts

// 设置工厂函数

const instanceFactory = (payload: Record<string, any>) => {

  const instance = context.container.get<InstanceClass>("Instance");

  instance.payload = payload;

  return instance;

};

// 工厂创建器,这里设置高阶函数的目的是将 context 传递给工厂函数,方便获取容器

const instanceFactoryCreator = (context: interfaces.Context) => {

  return instanceFactory;

};

// 绑定工厂

container

  .bind<interfaces.Factory<InstanceClass>>("Factory<InstanceClass>")

  .toFactory<InstanceClass>(instanceFactoryCreator);

```

在实例化后的运行时改变实例的属性,从而使实例中对属性的依赖得以满足。但这样的实现方式,会使得我们原有类的实现方式发生改变,也会改变类中属性的访问方式,例如原来时 readonly 或是 private 的属性,我们都无法在运行时对其进行赋值。

当这个类继承于外部需要传入参数的类,或者是需要在首次实例化时根据传入的变量依赖执行部分操作时,这种实例化的方式是行不通的。

那么如果我们的改造类具有以上特性,在不改变原有实现方式的情况下,应当如何做呢?

我们可以注意到,通过构造器注入的方式并不会将实例化时的行为交给容器,因此我们可以在这里进行手动的实例化并传入参数。那这样的实例化方式同样可以与工厂模式相结合,实现带参数实例化的工厂注入。

```ts

// 设置工厂函数

const instanceFactory = (payload: Record<string, any>) => {

  const InstanceClass = context.container.get<Newable<InstanceClass>>(

    "Newable<InstanceClass>"

  );

  const instance = new InstanceClass(payload);

  return instance;

};

```

注意,这里的 `new InstanceClass` 并不是引用原有类,而是引用了类的构造函数,而构造函数处于框架的管辖下,因此某种程度上该实例也是由框架进行了实例化得来的。因此,原有类甚至都不需要通过 `@injectable` 标注与注册。只需注册其构造器即可。

但始终,对于带参数实例化的工厂注入,它的实现方式并不优雅,也不符合依赖注入的思想。因此,本质上来说,类似于`类继承`的方式并不是一个好的`code smell`,我们推荐使用`对象组合`来代替`类继承`,从而规避掉需要在构造函数中为 super() 传入变量的尴尬局面。

## 结语

以上就是我在使用依赖注入框架重构项目时,对于多实例服务管理的一些思考与实践。它成功地帮我完成了整个项目的重构,也让我对于依赖注入框架有了更深的理解。

但于此同时,我也在实践中发现了许多依赖注入框架的局限性。但这并不说明依赖注入框架不够完善,而是说明了依赖注入作为一种设计模式与思想,它有其匹配的设计哲学。例如在上述的例子中,真正按照框架的最佳实践来说,我们应当只为服务注入行为抽象,而不是某些具体的变量数据,这对代码可测性来说非常重要。

因此,我更推荐在使用依赖注入框架前,先学习依赖注入的设计思想,再去使用框架。而不是尝试魔改某个依赖注入框架来迎合固有的编码风格。这不一定对设计与性能有正向的收益。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,233评论 6 495
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,357评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,831评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,313评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,417评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,470评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,482评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,265评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,708评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,997评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,176评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,503评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,150评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,391评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,034评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,063评论 2 352

推荐阅读更多精彩内容