新闻中心

J*aScript依赖注入容器

2025-10-17
浏览次数:
返回列表
依赖注入是通过外部注入依赖实现控制反转,提升解耦与可测试性;文中给出构造函数注入示例及简易DI容器实现,支持单例与瞬时生命周期管理,最后介绍使用场景与成熟库InversifyJS。

javascript依赖注入容器

J*aScript中的依赖注入(Dependency Injection, DI)容器是一种设计模式工具,用于管理对象之间的依赖关系,提升代码的可测试性、可维护性和解耦程度。在大型应用中,手动创建和管理依赖容易导致代码混乱,而DI容器能自动解析并注入所需依赖。

什么是依赖注入?

依赖注入的核心思想是:不主动在类内部创建依赖实例,而是由外部将依赖“注入”进来。这样可以实现控制反转(Inversion of Control, IoC),让模块之间更加松耦合。

常见的注入方式包括:

  • 构造函数注入:通过构造函数传入依赖
  • 属性注入:将依赖赋值给对象属性
  • 方法注入:通过方法参数传递依赖

为什么需要DI容器?

当项目变大,依赖层级加深时,手动管理依赖会变得繁琐且易错。DI容器的作用是:

迷你天猫商城 迷你天猫商城

迷你天猫商城是一个基于Spring Boot的综合性B2C电商平台,需求设计主要参考天猫商城的购物流程:用户从注册开始,到完成登录,浏览商品,加入购物车,进行下单,确认收货,评价等一系列操作。 作为迷你天猫商城的核心组成部分之一,天猫数据管理后台包含商品管理,订单管理,类别管理,用户管理和交易额统计等模块,实现了对整个商城的一站式管理和维护。所有页面均兼容IE10及以上现代浏览器。部署方式1、项目

迷你天猫商城 0 查看详情 迷你天猫商城
  • 自动解析依赖树,按需创建实例
  • 支持单例、瞬时等多种生命周期管理
  • 集中注册和配置服务,便于维护
  • 方便替换实现(如用mock替换真实服务进行测试)

一个简单的DI容器实现

下面是一个轻量级的DI容器示例,使用Map存储注册的服务,并支持构造函数注入:

class Container {
  constructor() {
    this.registry = new Map();
    this.instances = new Map(); // 缓存单例
  }

  register(name, ctor, lifecycle = 'singleton') {
    this.registry.set(name, { ctor, lifecycle });
    return this;
  }

  resolve(name) {
    if (!this.registry.has(name)) {
      throw new Error(`Service not registered: ${name}`);
    }

    const { ctor, lifecycle } = this.registry.get(name);

    if (lifecycle === 'singleton' && this.instances.has(name)) {
      return this.instances.get(name);
    }

    const instance = this._instantiate(ctor);
    
    if (lifecycle === 'singleton') {
      this.instances.set(name, instance);
    }

    return instance;
  }

  _instantiate(ctor) {
    const paramRegex = /constructor\s*\(\s*([^)]*)\)/;
    const argsStr = ctor.toString().match(paramRegex)?.[1] || '';
    const deps = argsStr.split(',').map(arg => arg.trim()).filter(Boolean);

    const dependencies = deps.map(dep => {
      if (!this.registry.has(dep)) {
        throw new Error(`Dependency not registered: ${dep}`);
      }
      return this.resolve(dep);
    });

    return Reflect.construct(ctor, dependencies);
  }
}

使用示例

假设我们有两个服务:Logger 和 UserService:

class Logger {
  log(msg) {
    console.log('[LOG]', msg);
  }
}

class UserService {
  constructor(Logger) {
    this.logger = Logger;
  }

  getUser(id) {
    this.logger.log(`Fetching user ${id}`);
    return { id, name: 'John Doe' };
  }
}

// 使用容器
const container = new Container();
container.register('Logger', Logger);
container.register('UserService', UserService);

const userService = container.resolve('UserService');
userService.getUser(1); // [LOG] Fetching user 1

基本上就这些。这个简易容器展示了DI的核心机制。实际项目中,也可以使用成熟的库,比如 InversifyJS,它支持装饰器、类型绑定和更复杂的场景,适合TypeScript项目。

以上就是J*aScript依赖注入容器的详细内容,更多请关注其它相关文章!


# java  # 关键词网站排名怎么看  # 重庆网站建设重点  # 营销推广ppt模板推荐夏天  # 上海贴心seo优化  # 保健品推广的营销方式  # 物流唐山网站优化方案  # seo关键词排名wl云速捷宀  # 相关文章  # 所需  # 是由  # 有哪些  # 是一种  # 如何实现  # 如何用  # 如何使用  # 是一个  # 可以使用  # red  # 为什么  # ai  # 工具  # typescript  # js  # javascript  # 独立网站怎么建设出来  # 江西seo营销怎么赚钱  # 忻州网站建设的优势 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: J*a中实现Go语言select通道多路复用机制  QQ邮箱网页版入口页面 QQ邮箱在线登录入口官网  高德地图怎么看全景照片_高德地图全景照片浏览教程  Mac怎么使用表情符号_Mac Emoji快捷键面板  电脑IP地址怎么查 查看本机IP地址的几种方法  Win11截图该按哪些键 Win11截屏完整流程解析【教程】  今日头条怎么同步内容到抖音_今日头条内容同步到抖音教程  知音漫客正版漫画平台_知音漫客官网账号登录  解决 Express.js 中 PUT 请求密码修改失败的路由配置指南  Go语言中对Map值调用带指针接收者方法:原理与最佳实践  深入理解字体排版:Adobe光学字偶距与CSS字偶距的差异与实现  百度浏览器字体显示异常偏小_百度浏览器字体渲染修复方案  sublime侧边栏怎么增强功能_SideBarEnhancements for sublime安装与配置  邮政编码查询不到怎么办_邮政编码查询不到的常见原因与对策  Composer中的^和~符号代表什么_精通Composer版本号语义化约束  反效果?《战地6》免费试玩开启后玩家数不升反降  Word2013如何插入视频和音频媒体_Word2013媒体插入的多媒体支持  Python自定义类排序:解决lambda键值访问TypeError的实践指南  HTML长属性值处理:表单action路径优化与代码规范应对  J*aScript数组对象转换:按指定键分组与值收集  12306选座怎么选到商务座_12306商务座选择与配置说明  PySpark中高效提取字符串右侧可变长度数字:使用regexp_extract  sublime如何优雅地处理行尾空格_sublime自动清理多余空白字符配置  vivo手机参数配置怎么增强信号_vivo手机参数配置信号增强方法  HTML转PPT成品工具有哪些?HTML网页转PPT成品工具大全  解决移动端滚动问题的overflow属性应用指南  如何提高微信支付的安全性_微信支付安全防护与设置建议  Composer如何处理Git子模块(submodule)依赖_Composer与Git Submodule的对比与选择  HTML元素状态管理:根据DIV内容动态启用/禁用按钮  C++如何操作大型数据集_使用C++流式处理(Streaming)技术避免一次性加载大文件  漫蛙2正版漫画站 漫蛙2网页版快速访问入口  outlook中文官网入口地址 outlook官方中文版直达首页链接  MongoDB聚合管道:正确匹配对象数组中_id的方法  QQ邮箱官方网站登录入口_QQ邮箱网页版在线使用  Win11文件资源管理器卡顿怎么修 Win11重置资源管理器进程优化响应速度【修复方法】  PostgreSQL海量数据高效导入策略:Python与Django实践指南  React Hooks最佳实践:动态组件状态管理的组件化方案  Spyder启动失败:字体文件权限拒绝错误解决方案  “在文档元素之后找到了标记”是什么错误? 检查并修复XML中多个根元素的3个方法  html5 app怎么运行环境_配html5 app运行环境【教程】  如何在更新Composer依赖后自动运行测试_使用post-update-cmd钩子触发PHPUnit  谷歌学术网站直达地址 谷歌学术搜索网页版一键进入  LINQ to XML为何解析失败? 深入理解C# XDocument的异常处理  12306选座系统怎么选连座_12306选座多人连坐操作方法  Fabric模组开发:自定义物品与物品组的现代管理方法  韩小圈电脑版在线入口_网页版免费登录地址  KFC游戏互动怎么赢取优惠券_KFC线上游戏活动参与与优惠代码赢取教程  CSS Flexbox如何实现多行排列_flex-wrap wrap自动换行显示  Golang如何测试channel通信行为_Golang channel通信测试与分析方法  正确连接J*aScript到HTML实现可点击图片与自定义事件处理 

搜索