我的编程空间,编程开发者的网络收藏夹
学习永远不晚

VSCode中的依赖注入怎么实现

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

VSCode中的依赖注入怎么实现

这篇文章主要讲解了“VSCode中的依赖注入怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“VSCode中的依赖注入怎么实现”吧!

依赖注入介绍

如果有这样一个模块 A,它的实现依赖另一个模块 B 的能力,那么应该如何设计呢?很简单,我们可以在 A 模块的构造函数中实例化模块 B,这样就可以在模块 A 内部使用模块 B 的能力了。

class A {  constructor() {    this.b = new B();  }}class B {}const a = new A();

但是这样做有两个问题,一是模块 A 的实例化过程中,需要手动实例化模块 B,而且如果模块 B 的依赖关系发生变化,那么也需要修改模块 A 的构造函数,导致代码耦合。

二是在复杂项目中,我们在实例化模块 A 时,难以判断模块 B 是否被其他模块依赖而已经实例化过了,从而可能将模块 B 多次实例化。若模块 B 较重或者需要为单例设计,这将带来性能问题。

因此,更好的方式是,将所有模块的实例化交给外层框架,由框架统一管理模块的实例化过程,这样就可以解决上述两个问题。

class A {  constructor(private b: B) {    this.b = b;  }}class B {}class C {  constructor(private a: A, private b: B) {    this.b = b;  }}const b = new B();const a = new A(b);const c = new C(a, b);

这种将依赖对象通过外部注入,避免在模块内部实例化依赖的方式,称为依赖注入 (Dependencies Inject, 简称 DI)。这在软件工程中是一种常见的设计模式,我们在 Java 的 Spring,JS 的 Angular,Node 的 NestJS 等框架中都可以看到这种设计模式的应用。

当然,在实际应用中,由于模块众多,依赖复杂,我们很难像上面的例子一样,规划出来每个模块的实例化时机,从而编写模块实例化顺序。并且,许多模块可能并不需要第一时间被创建,需要按需实例化,因此,粗暴的统一实例化是不可取的。

因此我们需要一个统一的框架来分析并管理所有模块的实例化过程,这就是依赖注入框架的作用。

借助于 TypeScript 的装饰器能力,VSCode 实现了一个极为轻量化的依赖注入框架。我们可以先来简单实现一下,解开这个巧妙设计的神秘面纱。

最简依赖注入框架设计

实现一个依赖注入框架只需要两步,一个是将模块声明并注册到框架中进行管理,另一个是在模块构造函数中,声明所需要依赖的模块有哪些。

我们先来看模块的注册过程,这需要 TypeScript 的类装饰器能力。我们在注入时,只需要判断模块是否已经注册,如果没有注册,将模块的 id(这里简化为模块 Class 名称)与类型传入即可完成单个模块的注册。

export function Injectable(): ClassDecorator {  return (Target: Class): any => {    if (!collection.providers.has(Target.name)) {      collection.providers.set(Target.name, target);    }    return target;  };}

之后我们再来看看模块是如何声明依赖的,这需要 TypeScript 的属性装饰器能力。我们在注入时,先判断依赖的模块是否已经被实例化,如果没有,则将依赖模块进行实例化,并存入框架中管理。最终返回已经被实例化完成的模块实例。

export function Inject(): PropertyDecorator {  return (target: Property, propertyKey: string) => {    const instance = collection.dependencies.get(propertyKey);    if (!instance) {      const DependencyProvider: Class = collection.providers.get(propertyKey);      collection.dependencies.set(propertyKey, new DependencyProvider());    }    target[propertyKey] = collection.dependencies.get(propertyKey);  };}

最后只需要保证框架本身在项目运行前完成实例化即可。(在例子中表示为 injector)

export class ServiceCollection {  readonly providers = new Map<string, any>();  readonly dependencies = new Map<string, any>();}const collection = new ServiceCollection();export default collection;

这样,一个最简化的依赖注入框架就完成了。由于保存了模块的类型与实例,它实现了模块的按需实例化,无需在项目启动时就初始化所有模块。

我们可以尝试调用它,以上面举出的例子为例:

@injectable()class A {  constructor(@inject() private b: B) {    this.b = b;  }}@injectable()class B {}class C {  constructor(@inject() private a: A, @inject() private b: B) {    this.b = b;  }}const c = new C();

无需知晓模块 A,B 的实例化时机,直接初始化任何一个模块,框架会自动帮你找到并实例化好所有依赖的模块。

VSCode 的依赖收集实现

上面介绍了一个依赖注入框架的最简实现。但当我们真正阅读 VSCode 的源码时,我们发现 VSCode 中的依赖注入框架貌似并不是这样消费的。

例如在下面这段鉴权服务中,我们发现该类并没有@injectable()作为类的依赖收集,并且依赖服务也直接用其类名作为修饰器,而不是@inject()

// class="lazy" data-src\vs\workbench\services\authentication\browser\authenticationService.tsexport class AuthenticationService extends Disposable implements IAuthenticationService {  constructor(    @IActivityService private readonly activityService: IActivityService,    @IExtensionService private readonly extensionService: IExtensionService,    @IStorageService private readonly storageService: IStorageService,    @IRemoteAgentService private readonly remoteAgentService: IRemoteAgentService,    @IDialogService private readonly dialogService: IDialogService,    @IQuickInputService private readonly quickInputService: IQuickInputService  ) {}}

其实这里的修饰符并不是真正指向类名,而是一个同名的资源描述符 id(VSCode 中称之为 ServiceIdentifier),通常使用字符串或 Symbol 标识。

通过 ServiceIdentifier 作为 id,而不是简单粗暴地通过类名称作为 id 注册 Service,有利于处理项目中一个 interface 可能存在多态实现,需要同时多个同名类实例的问题。

此外,在构造 ServiceIdentifier 时,我们便可以将该类声明注入框架,而无需@injectable()显示调用了。

那么,这样一个 ServiceIdentifier 该如何构造呢?

// class="lazy" data-src\vs\platform\instantiation\common\instantiation.tsexport function createDecorator<T>(serviceId: string): ServiceIdentifier<T> {  if (_util.serviceIds.has(serviceId)) {    return _util.serviceIds.get(serviceId)!;  }  const id = <any>function (target: Function, key: string, index: number): any {    if (arguments.length !== 3) {      throw new Error('@IServiceName-decorator can only be used to decorate a parameter');    }    storeServiceDependency(id, target, index);  };  id.toString = () => serviceId;  _util.serviceIds.set(serviceId, id);  return id;}// 被 ServiceIdentifier 装饰的类在运行时,将收集该类的依赖,注入到框架中。function storeServiceDependency(id: Function, target: Function, index: number): void {  if ((target as any)[_util.DI_TARGET] === target) {    (target as any)[_util.DI_DEPENDENCIES].push({ id, index });  } else {    (target as any)[_util.DI_DEPENDENCIES] = [{ id, index }];    (target as any)[_util.DI_TARGET] = target;  }}

我们仅需通过createDecorator方法为类创建一个唯一的ServiceIdentifier,并将其作为修饰符即可。

以上面的 AuthenticationService 为例,若所依赖的 ActivityService 需要变更多态实现,仅需修改 ServiceIdentifier 修饰符确定实现方式即可,无需更改业务的调用代码。

export const IActivityServicePlanA = createDecorator<IActivityService>("IActivityServicePlanA");export const IActivityServicePlanB = createDecorator<IActivityService>("IActivityServicePlanB");export interface IActivityService {...}export class AuthenticationService {  constructor(    @IActivityServicePlanA private readonly activityService: IActivityService,  ) {}}

循环依赖问题

模块之间的依赖关系是有可能存在循环依赖的,比如 A 依赖 B,B 依赖 A。这种情况下进行两个模块的实例化会造成死循环,因此我们需要在框架中加入循环依赖检测机制来进行规避。

本质上,一个健康的模块依赖关系就是一个有向无环图(DAG),我们之前介绍过有向无环图在 excel 表格函数中的应用,放在依赖注入框架的设计中也同样适用。

我们可以通过深度优先搜索(DFS)来检测模块之间的依赖关系,如果发现存在循环依赖,则抛出异常。

// class="lazy" data-src/vs/platform/instantiation/common/instantiationService.tswhile (true) {  let roots = graph.roots();  // if there is no more roots but still  // nodes in the graph we have a cycle  if (roots.length === 0) {    if (graph.length !== 0) {      throwCycleError();    }    break;  }  for (let root of roots) {    // create instance and overwrite the service collections    const instance = this._createInstance(root.data.desc, []);    this._services.set(root.data.id, instance);    graph.removeNode(root.data);  }}

该方法通过获取图节点的出度,将该类的全部依赖提取出来作为roots,然后逐个实例化,并从途中剥离该依赖节点。由于依赖树的构建是逐层依赖的,因此按顺序实例化即可。当发现该类的所有依赖都被实例化后,图中仍存在节点,则认为存在循环依赖,抛出异常。

感谢各位的阅读,以上就是“VSCode中的依赖注入怎么实现”的内容了,经过本文的学习后,相信大家对VSCode中的依赖注入怎么实现这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

VSCode中的依赖注入怎么实现

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

VSCode中的依赖注入怎么实现

这篇文章主要讲解了“VSCode中的依赖注入怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“VSCode中的依赖注入怎么实现”吧!依赖注入介绍如果有这样一个模块 A,它的实现依赖另一个
2023-07-04

VSCode中依赖注入的原理是什么

这篇文章主要介绍“VSCode中依赖注入的原理是什么”,在日常操作中,相信很多人在VSCode中依赖注入的原理是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”VSCode中依赖注入的原理是什么”的疑惑有所
2023-07-05

Laravel的依赖注入怎么实现

这篇文章主要介绍了Laravel的依赖注入怎么实现的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Laravel的依赖注入怎么实现文章都会有所收获,下面我们一起来看看吧。在Laravel中,依赖注入的实现原理是利
2023-06-29

python依赖注入怎么实现

在Python中实现依赖注入有多种方法,下面介绍两种常用的实现方式:1. 使用第三方库:可以使用流行的依赖注入框架,例如Dagger、Inject等。这些框架提供了注解和装饰器等方式来标记依赖项,并自动解析和注入依赖项。例如,使用Dagge
2023-08-30

golang依赖注入怎么实现

在Go语言中实现依赖注入可以使用两种常见的方式:构造函数注入和接口注入。1. 构造函数注入:构造函数注入是通过在结构体中定义一个包含所需依赖的构造函数来实现的。例如:```gotype Database struct {// ...}fun
2023-09-17

PHP中怎样实现依赖注入

今天就跟大家聊聊有关PHP中怎样实现依赖注入,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。我们来看一段代码:class A{ public function echo() { ech
2023-06-04

spring怎么实现依赖注入DI

这篇文章主要介绍了spring怎么实现依赖注入DI的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇spring怎么实现依赖注入DI文章都会有所收获,下面我们一起来看看吧。spring依赖注入DI1、创建一个mav
2023-06-29

SpringBoot中如何实现注入依赖

SpringBoot中如何实现注入依赖?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。今天给大家介绍一下SpringBoot中是如何实现依赖注入的功能。在以往spring使用
2023-05-31

C#中如何实现依赖注入

在C#中实现依赖注入通常可以使用一些现成的框架,比如ASP.NET Core中自带的依赖注入容器。以下是一个简单的示例:首先,定义一个接口和一个实现该接口的类:public interface IMyService{void DoSo
C#中如何实现依赖注入
2024-04-03

ASP.NET Core依赖关系注入怎么实现

本篇内容主要讲解“ASP.NET Core依赖关系注入怎么实现”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“ASP.NET Core依赖关系注入怎么实现”吧!1.前言面向对象设计(OOD)里有一
2023-06-29

AndroidHilt依赖注入的实现浅析

依赖注入是一个听起来很“高大上”的概念,但是其实很简单。本来我要接受各种参数自己构造一个对象,现在只接受一个已经实例化的对象直接作为参数
2023-01-28

java Spring中如何实现依赖注入

总得来说,Spring中依赖注入有这么三种方式:①通过set方式赋值②通过构造器赋值③通过p标签赋值java相关视频教程推荐:java视频下面我们稍稍展开来谈:1.通过set方式赋值主要用到围堵标签来传递值此处的name就是Teacher类里声明的变量名,va
java Spring中如何实现依赖注入
2019-11-15

如何在PHP中实现依赖注入

这篇文章给大家介绍如何在PHP中实现依赖注入,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。首先,我们来看一段代码:class A{ public function echo() {
2023-06-17

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录