使用JavaScript实现一个简单的依赖注入_js架构

答案:依赖注入容器通过外部传入依赖实现解耦,JavaScript 实现包含注册、解析功能,支持单例与嵌套依赖,可管理类或函数服务。

使用javascript实现一个简单的依赖注入_js架构

依赖注入(Dependency Injection, DI)是一种设计模式,用于实现控制反转(IoC),让对象的依赖由外部传入,而不是在内部创建。这种方式提高了代码的可测试性、灵活性和解耦程度。下面用 JavaScript 实现一个简单的依赖注入容器。

1. 实现一个基础的依赖注入容器

我们创建一个 Container 类,用来注册、解析和管理依赖。

class Container {
  constructor() {
    this.dependencies = {};
  }

  // 注册依赖
  register(name, dependency, singleton = false) {
    this.dependencies[name] = {
      factory: typeof dependency === ‘function’ ? dependency : () => dependency,
      singleton,
      instance: null
    };
  }

  // 解析依赖(获取实例)
  resolve(name) {
    const dep = this.dependencies[name];
    if (!dep) {
      throw new Error(`Dependency not found: ${name}`);
    }

    if (dep.singleton && dep.instance) {
      return dep.instance;
    }

    const instance = dep.factory(this); // 把 container 传进去,支持依赖嵌套
    if (dep.singleton) {
      dep.instance = instance;
    }
    return instance;
  }

  // 清除某个依赖或全部依赖(用于测试)
  clear(name) {
    if (name) {
      delete this.dependencies[name];
    } else {
      this.dependencies = {};
    }
  }
}

2. 定义服务类

我们定义几个简单的服务来演示依赖关系。

// 日志服务
function Logger() {
  return {
    log: (msg) => console.log(`[LOG] ${msg}`)
  };
}

// 数据服务,依赖 Logger
function DataService(container) {
  const logger = container.resolve(‘logger’);
  return {
    getData: () => {
      logger.log(‘Fetching data…’);
      return [‘item1’, ‘item2’];
    }
  };
}

// 用户服务,依赖 DataService
function UserService(container) {
  const dataService = container.resolve(‘dataService’);
  const logger = container.resolve(‘logger’);
  return {
    getUserData: () => {
      logger.log(‘Getting user data’);
      return dataService.getData();
    }
  };
}

3. 使用容器注册并解析依赖

现在我们使用容器来注册服务,并获取实例。

立即学习“Java免费学习笔记(深入)”;

const container = new Container();

container.register(‘logger’, Logger, true);
container.register(‘dataService’, DataService, true);
container.register(‘userService’, UserService, true);

// 解析顶层服务,自动触发依赖加载
const userService = container.resolve(‘userService’);
console.log(userService.getUserData());
// 输出:
// [LOG] Getting user data
// [LOG] Fetching data…
// [‘item1’, ‘item2’]

4. 支持构造函数或 class 的写法

如果使用 class,也可以适配。只需要 factory 返回 new 实例即可。

class ApiClient {
  constructor(logger) {
    this.logger = logger;
  }
  fetch() {
    this.logger.log(‘API request sent’);
    return { success: true };
  }
}

// 注册时传入工厂函数
container.register(‘apiClient’, (container) => {
  const logger = container.resolve(‘logger’);
  return new ApiClient(logger);
}, true);

基本上就这些。这个简易的 DI 容器支持:

注册值、函数、对象 单例模式控制 依赖自动解析与传递 container 上下文 支持 class 和 function 构造的服务

不复杂但容易忽略的是:把 container 作为参数传给 factory,这样内部才能 resolve 其他依赖。

以上就是使用JavaScript实现一个简单的依赖注入_js架构的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1535347.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月21日 02:21:55
下一篇 2025年12月21日 02:22:09

相关推荐

  • JS函数如何定义函数调试技巧_JS函数调试技巧定义与断点设置方法

    JS函数可通过函数声明、表达式、箭头函数和构造函数定义;2. 调试时可用开发者工具设断点或插入debugger语句暂停执行,便于检查状态。 在JavaScript开发中,函数是程序的核心组成部分。掌握如何定义函数以及有效的调试技巧,能大幅提升开发效率和代码质量。下面介绍JS函数的定义方式以及实用的调…

    2025年12月21日
    000
  • JavaScript RESTful API设计与实现

    答案:使用Node.%ignore_a_1%和Express可快速构建RESTful API,通过GET、POST、PUT、DELETE操作实现用户资源的增删改查,结合路由模块化、统一响应格式、输入验证与错误处理提升API质量,确保语义清晰、结构规范、易于维护。 在现代Web开发中,JavaScri…

    2025年12月21日
    000
  • js垃圾回收的场景优化

    JavaScript垃圾回收通过标记-清除算法自动释放无用对象,优化需及时断开无效引用。1. 解除事件监听与定时器避免残留回调;2. 闭包中避免长期持有大对象,使用后置null;3. 移除DOM后清除引用,缓存用WeakMap/WeakSet防泄漏;4. WeakMap键对象可被回收,适合元数据存储…

    2025年12月21日
    000
  • js数组中filter方法的使用

    filter()方法创建新数组包含满足条件的元素,体现不可变性;语法为arr.filter(callback, thisArg),callback接收element、index、array参数;常用于筛选数字、对象属性或去重;可与map、sort等链式调用处理数据,如先过滤成人用户再提取姓名;原数组…

    2025年12月21日
    000
  • JavaScript中的URLSearchParams对象使用_javascript网络请求

    URLSearchParams是JavaScript中用于处理URL查询参数的内置对象,可解析、构造、修改和遍历查询字符串。支持通过查询字符串、键值对对象或当前URL创建实例,自动编码确保合法性。提供get、getAll、append、set、delete、has和toString等方法,便于动态操…

    2025年12月21日
    000
  • JavaScript轮播图导航箭头点击无响应问题排查与修复

    本教程旨在解决javascript轮播图导航箭头点击失效的问题。核心原因在于js代码中用于事件监听的buttons变量未正确定义,导致foreach方法无法执行。文章将详细指导如何正确选取dom中的按钮元素,并将返回的类数组对象转换为标准数组,以便成功绑定点击事件,从而实现轮播图的正常导航功能。 在…

    2025年12月21日
    000
  • 在 React Router 中利用 Outlet 实现组件嵌套渲染

    对应的完整路径是 /admin/dashboard。同样,<route path="add-product" element="{} /> 对应的完整路径是 /admin/add-product。adminRoute 的处理: 在 App.js 中,如果你希…

    2025年12月21日
    000
  • 使用JavaScript实现一个简单的日历组件_javascript UI组件

    答案:该日历组件通过原生JavaScript实现,包含年月切换、当月日期生成及跨年处理。结构上分为头部按钮、标题和主体表格,支持动态渲染并补全首尾空白单元格,结合CSS美化样式,可嵌入网页展示当前月份日期,逻辑清晰且便于扩展功能。 实现一个简单的日历组件,可以展示当前月份的日期,并支持切换年份和月份…

    2025年12月21日
    000
  • JS数组如何映射_JavaScript数组map方法使用与转换数据案例

    JavaScript数组的map方法用于创建一个新数组,其结果是对原数组每个元素调用提供的函数后返回的值。它不会修改原始数组,而是返回一个经过转换的新数组。该方法常用于数据转换、界面渲染或准备接口数据。基本语法为array.map(function(currentValue, index, arr)…

    2025年12月21日
    000
  • 利用React Router v6的Outlet实现复杂布局的组件嵌套

    :当访问/admin/dashboard时,AdminMain组件将渲染到Dashboard中的Outlet位置。<route path="add-product" element="{} />:当访问/admin/add-product时,AddProdu…

    2025年12月21日
    000
  • JavaScript中的内存泄漏与排查方法_javascript性能优化

    内存泄漏在JavaScript单页应用中可能导致性能下降或崩溃。常见类型包括:意外的全局变量、未解绑的事件监听器、闭包引用、定时器回调和DOM引用未释放。使用Chrome DevTools的堆快照、内存分配时间线、Detached DOM树和Performance面板可定位泄漏。预防策略有:避免全局…

    2025年12月21日
    000
  • Canvas像素级图像替换的性能优化指南

    本文旨在解决html canvas中大规模像素级图像替换的性能瓶颈和潜在的图像缩放问题。通过引入离屏canvas技术和直接操作像素数据(`getimagedata`与`putimagedata`),我们将展示如何高效、准确地将特定颜色区域替换为另一张图像的纹理,显著提升渲染性能并避免不必要的`dra…

    2025年12月21日
    000
  • js脚本怎么获取当前页面滚动位置_js滚动位置获取脚本编写与实例

    获取页面滚动位置主要用window.scrollY和scrollX或兼容写法。通过getScrollTop()和getScrollLeft()函数可获取垂直和水平滚动距离,结合onscroll事件实时监听,用于显示回到顶部按钮或判断是否滚动到底部实现无限加载,需注意兼容性与性能优化。 JavaScr…

    2025年12月21日
    000
  • 利用 React Router Outlet 实现组件嵌套渲染与布局管理

    :这是一个子路由。当 URL 为 /admin/dashboard 时,AdminMain 组件将会在 Dashboard 组件内部的 位置渲染。<route path="add-product" element="{} />:同理,当 URL 为 /adm…

    2025年12月21日
    000
  • 前端图片压缩与处理的JavaScript方案_js性能优化

    前端图片压缩通过Canvas和File API在浏览器端实现,利用toBlob设置质量参数减小体积,结合URL.createObjectURL预览,使用Web Worker避免阻塞,推荐CompressorJS等库处理兼容性问题,优先采用WebP格式并控制压缩阈值,平衡画质与性能。 前端图片压缩与处…

    2025年12月21日
    000
  • Next.js App Router 中服务器组件的类型定义与最佳实践

    本文旨在指导开发者在 next.js 13+ 的 app router 架构下,如何正确地为服务器组件(包括页面组件和通用组件)进行类型定义。文章将解释 `nextpage` 类型不再适用于 app router 的原因,并提供针对 `page.tsx` 文件中异步服务器组件以及其他通用服务器组件的…

    2025年12月21日
    000
  • Next.js 服务器组件的类型声明指南

    本文深入探讨了next.js 13及更高版本中服务器组件的正确类型定义方法。我们将明确区分旧版`pages`目录中`nextpage`的用法,并重点介绍`app`目录中页面组件和通用服务器组件的类型实践,包括如何处理异步操作、组件参数以及利用typescript的类型推断功能,以提升代码的健壮性和可…

    2025年12月21日 好文分享
    000
  • 理解JavaScript中的模块联邦_javascript模块化

    模块联邦是Webpack 5实现微前端的核心技术,允许应用在运行时动态共享模块。通过ModuleFederationPlugin配置远程应用暴露模块、宿主应用按需加载,实现独立部署与构建。其核心优势在于支持独立开发部署、减少重复打包、提升加载效率,适用于多团队协作的大型系统如中台或电商平台。关键在于…

    2025年12月21日
    000
  • JavaScript中操作二进制数据的方法_javascript高级

    JavaScript通过ArrayBuffer、TypedArray和DataView实现二进制数据操作:ArrayBuffer为固定长度的原始二进制数据缓冲区,需通过视图访问;TypedArray(如Uint8Array、Float32Array)提供带类型的数据视图,适用于图像、音频等场景;Da…

    2025年12月21日
    000
  • JS实现前端权限路由控制_javascript实战

    答案:前端权限路由通过用户角色动态生成可访问路由。登录后,后端返回角色权限,前端根据meta字段中的角色信息过滤动态路由,利用router.addRoute注册合法路由,并结合导航守卫控制跳转,确保用户仅能访问授权页面,同时菜单同步渲染,提升安全与体验。 前端权限路由控制是现代单页应用中常见的需求,…

    2025年12月21日
    000

发表回复

登录后才能评论
关注微信