分解依赖倒置、IoC 和 DI

分解依赖倒置、ioc 和 di

本文深入探讨 NestJS 依赖注入系统,并阐明依赖倒置原则 (DIP)、控制反转 (IoC) 和依赖注入 (DI) 的概念及其关联。这三个概念看似相似,实则各有侧重,相互关联却又解决不同的问题。本文旨在帮助读者理清这些概念,并理解它们如何协同工作。

依赖倒置原则 (DIP)

定义:

高层模块不应该依赖于低层模块;两者都应该依赖于抽象。抽象不应该依赖于细节;细节应该依赖于抽象。

含义解读

在软件开发中,高层模块负责核心业务逻辑,而低层模块处理具体的实现细节(例如文件系统、数据库或 API)。若无 DIP,高层模块直接依赖于低层模块,导致紧密耦合,从而:

降低灵活性。复杂化测试和维护。难以替换或扩展低层细节。

DIP 颠覆了这种依赖关系。高层和低层模块都依赖于共享的抽象(接口或抽象类),而不是高层模块直接控制低层实现。

无 DIP 示例

Python 示例

class EmailService:    def send_email(self, message):        print(f"Sending email: {message}")class Notification:    def __init__(self):        self.email_service = EmailService()    def notify(self, message):        self.email_service.send_email(message)

TypeScript 示例

class EmailService {    sendEmail(message: string): void {        console.log(`Sending email: ${message}`);    }}class Notification {    private emailService: EmailService;    constructor() {        this.emailService = new EmailService();    }    notify(message: string): void {        this.emailService.sendEmail(message);    }}

问题:

紧密耦合:Notification 直接依赖 EmailService难以扩展:切换到 SMSServicePushNotificationService 需要修改 Notification 类。

含 DIP 示例

Python 示例

from abc import ABC, abstractmethodclass MessageService(ABC):    @abstractmethod    def send_message(self, message):        passclass EmailService(MessageService):    def send_message(self, message):        print(f"Sending email: {message}")class Notification:    def __init__(self, message_service: MessageService):        self.message_service = message_service    def notify(self, message):        self.message_service.send_message(message)# 使用示例email_service = EmailService()notification = Notification(email_service)notification.notify("Hello, Dependency Inversion!")

TypeScript 示例

interface MessageService {    sendMessage(message: string): void;}class EmailService implements MessageService {    sendMessage(message: string): void {        console.log(`Sending email: ${message}`);    }}class Notification {    private messageService: MessageService;    constructor(messageService: MessageService) {        this.messageService = messageService;    }    notify(message: string): void {        this.messageService.sendMessage(message);    }}// 使用示例const emailService = new EmailService();const notification = new Notification(emailService);notification.notify("Hello, Dependency Inversion!");

DIP 的优势

灵活性:无需修改高层模块即可替换实现。可测试性:使用模拟对象替换真实依赖项进行测试。可维护性:低层模块的更改不会影响高层模块。

控制反转 (IoC)

IoC 是一种设计原则,依赖项的创建和管理由外部系统或框架控制,而不是在类内部进行。

传统编程中,类自行创建和管理依赖项。IoC 将这种控制反转——外部实体(例如框架或容器)管理依赖项,并在需要时注入。

无 IoC 示例

在无 IoC 的场景中,类自行创建和管理依赖项,导致紧密耦合。

Python 示例:无 IoC

class SMSService:    def send_message(self, message):        print(f"Sending SMS: {message}")class Notification:    def __init__(self):        self.sms_service = SMSService()    def notify(self, message):        self.sms_service.send_message(message)# 使用示例notification = Notification()notification.notify("Hello, tightly coupled dependencies!")

TypeScript 示例:无 IoC

class SMSService {    sendMessage(message: string): void {        console.log(`Sending SMS: ${message}`);    }}class Notification {    private smsService: SMSService;    constructor() {        this.smsService = new SMSService();    }    notify(message: string): void {        this.smsService.sendMessage(message);    }}// 使用示例const notification = new Notification();notification.notify("Hello, tightly coupled dependencies!");

无 IoC 的问题:

紧密耦合:Notification 类直接创建并依赖 SMSService 类。灵活性低:切换到其他实现需要修改 Notification 类。难以测试:模拟依赖项用于单元测试比较困难,因为依赖项是硬编码的。

使用 IoC

在使用 IoC 的示例中,依赖关系的管理责任转移到外部系统或框架,实现松散耦合并增强可测试性。

Python 示例:使用 IoC

class SMSService:    def send_message(self, message):        print(f"Sending SMS: {message}")class Notification:    def __init__(self, message_service):        self.message_service = message_service    def notify(self, message):        self.message_service.send_message(message)# IoC: 依赖项由外部控制sms_service = SMSService()notification = Notification(sms_service)notification.notify("Hello, Inversion of Control!")

TypeScript 示例:使用 IoC

class SMSService {    sendMessage(message: string): void {        console.log(`Sending SMS: ${message}`);    }}class Notification {    private messageService: SMSService;    constructor(messageService: SMSService) {        this.messageService = messageService;    }    notify(message: string): void {        this.messageService.sendMessage(message);    }}// IoC: 依赖项由外部控制const smsService = new SMSService();const notification = new Notification(smsService);notification.notify("Hello, Inversion of Control!");

IoC 的优势:

松散耦合:类不创建其依赖项,从而减少对特定实现的依赖。易于切换实现:用 EmailService 替换 SMSService 而无需修改核心类。提高可测试性:在测试期间注入模拟或 Mock 依赖项。

依赖注入 (DI)

DI 是一种技术,对象从外部源接收其依赖项,而不是自己创建它们。

DI 是 IoC 的一种实现方式。它允许开发人员通过多种方式将依赖项“注入”到类中:

构造函数注入:依赖项通过构造函数传递。Setter 注入:依赖项通过公共方法设置。接口注入:依赖项通过接口提供。

Python 示例:使用 DI 框架

使用 injector 库:

from injector import Injector, injectclass EmailService:    def send_message(self, message):        print(f"Email sent: {message}")class Notification:    @inject    def __init__(self, email_service: EmailService):        self.email_service = email_service    def notify(self, message):        self.email_service.send_message(message)# DI 容器injector = Injector()notification = injector.get(Notification)notification.notify("This is Dependency Injection in Python!")

TypeScript 示例:使用 DI 框架

使用 tsyringe

import "reflect-metadata";import { injectable, inject, container } from "tsyringe";@injectable()class EmailService {    sendMessage(message: string): void {        console.log(`Email sent: ${message}`);    }}@injectable()class Notification {    constructor(@inject(EmailService) private emailService: EmailService) {}    notify(message: string): void {        this.emailService.sendMessage(message);    }}// DI 容器const notification = container.resolve(Notification);notification.notify("This is Dependency Injection in TypeScript!");

DI 的优势:

简化测试:轻松使用模拟对象替换依赖项。提高可扩展性:添加新的实现而无需修改现有代码。增强可维护性:减少系统某一部分变更的影响。

以上就是分解依赖倒置、IoC 和 DI的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月13日 19:23:26
下一篇 2025年12月11日 15:28:03

相关推荐

  • 使用 LangChain 和 OpenAI 构建智能代理:开发人员指南

    人工智能技术日新月异,开发者们正积极探索将智能功能融入日常工作流程的方法。构建能够自主完成任务、将推理与行动相结合的智能代理便是其中一种有效途径。本文将指导您如何利用LangChain、OpenAI的GPT-4以及LangChain的实验工具,创建一个能够执行Python代码、处理CSV文件并解答复…

    2025年12月13日
    000
  • 关于 Python 的 5 件事

    Python 持续高速发展,每个新版本都带来诸多改进和新功能。2025 年,Python 开发者将迎来一系列令人振奋的更新,旨在提升性能、代码可读性和整体开发体验。以下是一些亮点: 结构化模式匹配的完善 (PEP 634): 在 Python 3.10 首次引入,并在 Python 3.11 中得到…

    2025年12月13日
    000
  • 自动化你的求职:使用 Python 抓取 + LinkedIn 职位

    linkedin 数据显示,求职者平均每周花费 11 小时寻找工作。技术职位竞争更激烈,需要在多个平台筛选数百个职位。我的伴侣找工作时,每天都要花数小时浏览 linkedin,这促使我寻找更有效的方法。 挑战 对于 Web 开发人员而言,就业市场竞争激烈。在伦敦搜索“前端开发人员”,结果多达 401…

    2025年12月13日
    000
  • 为什么编码训练营让开发人员失败

    几年前,编码训练营被吹捧为进入科技行业的捷径。经过密集的培训,一系列项目,你就能找到软件工程师的工作——这是当时的承诺。 然而,如今,这一承诺正受到质疑。 并非说编码训练营完全无效——一些学员确实成功就业了。但现实是,许多毕业生发现训练营并没能帮助他们找到工作。考虑到开发人员的裁员潮、初级开发人员的…

    2025年12月13日
    000
  • Agentic AI:构建生产级、企业级 AI

    在蓬勃发展的人工智能领域,Agentic AI 正在成为大型企业数字化转型的核心驱动力。不同于只能执行预设指令的传统AI系统,Agentic AI 具备自主行动、动态学习和解决复杂问题的能力,无需持续的人工干预。这代表着一种全新的范式转变,帮助企业优化运营流程、提升决策效率,并最终提供卓越的客户体验…

    2025年12月13日
    000
  • 如何将开源 Python 项目变成赚钱机器

    想象一下:您是一位充满热情的开发人员,在一个解决实际问题的开源 python 项目上熬夜。你把它释放到野外,它就会获得牵引力。人们正在使用它、喜欢它并赞扬你的工作。但问题是——你不会从中赚到一分钱。这听起来很熟悉吗?如果我告诉你有一种方法可以将你的热情转化为利润,而又不会出卖或损害你的价值观,你会怎…

    2025年12月13日
    000
  • 在没有硬编码 ID 的情况下使用带有外键的 Django Fixture

    Django Fixtures 提供了一种便捷的方式将示例数据加载到数据库中。然而,如果外键 ID 采用硬编码方式,当 ID 发生变化或数据在数据库间迁移时,Fixture 可能会失效。 更好的方案是使用自然键,它允许您通过有意义的值(而非数字 ID)来引用外键。 避免在 Fixture 中硬编码 …

    2025年12月13日
    000
  • PyTorch 中的随机透视

    请我喝杯咖啡☕ *备忘录: 我的帖子解释了 randomrotation()。我的帖子解释了 randomaffine()。我的帖子解释了 randomhorizo​​ntalflip()。我的帖子解释了 randomverticalflip()。我的帖子解释了 oxfordiiitpet()。 r…

    2025年12月13日 好文分享
    000
  • 使用 AWS 和 OpenWeatherMap API 构建天气数据分析管道

    大家好!本文将指导您构建一个利用openweathermap api和aws服务进行天气数据分析的完整数据管道。该项目涵盖数据获取、s3存储、aws glue数据编目以及amazon athena查询等步骤,实现可扩展高效的天气数据处理。 项目概述 本项目旨在创建一个可扩展、高效的数据管道,用于收集…

    2025年12月13日 好文分享
    000
  • 可扩展的 Python 后端:使用 uv、Docker 和预提交构建容器化 FastAPI 应用程序:分步指南

    在当今容器化部署的世界中,高效构建和部署后端应用程序至关重要。 fastapi 已成为创建快速、高性能 api 的最流行的 python 框架之一。为了管理依赖关系,我们还可以利用 uv(包管理器)作为一个方便的工具。 紫外线 我假设您之前已经在本地安装了 uv 和 docker。 现在,我们可以通…

    2025年12月13日
    000
  • 使用 Anthropic 的 Claude Sonnet 生成报告

    Pilar,一家巴西房地产科技公司,联合创始人兼首席技术官Raphael分享了利用Anthropic Claude 3.5 Sonnet生成报告的经验,并比较了两种不同方法的优劣。Pilar为房地产经纪商提供基于低成功费模式的软件和服务,其20人的技术团队不断开发创新产品,例如全新的房地产门户网站P…

    2025年12月13日
    000
  • Python 垃圾收集:您需要了解的一切

    一、Python垃圾回收机制详解 在计算机领域,垃圾回收(Garbage Collection, GC)是自动内存管理的关键技术,它负责回收程序不再使用的内存空间。这项技术极大地减轻了程序员的负担,降低了内存泄漏的风险。许多现代编程语言,例如Python、Java、Go等,都内置了垃圾回收机制。Py…

    2025年12月13日 好文分享
    000
  • 使用 SPython 和 OpenWeather API 构建天气仪表板

    本项目是一个基于python的应用程序,用于获取和显示openweather api提供的实时天气数据。它展示了如何使用第三方api、处理http请求、解析json响应以及以用户友好的方式呈现天气信息。 主要功能: 获取指定位置的实时天气数据。显示详细天气信息,包括温度、湿度、风速和天气状况。自动将…

    2025年12月13日 好文分享
    000
  • 使用 Python 的密码生成脚本

    Python 密码生成器:轻松创建强密码 本文将指导您使用 python 创建一个简单的密码生成器,帮助您生成安全可靠的随机密码,保护您的在线账户安全。 python 的灵活性和内置功能使其成为实现此工具的理想选择。 强密码通常包含大小写字母、数字和特殊字符的组合。 密码生成器关键特性: 随机性: …

    2025年12月13日
    000
  • 构建人工智能销售代理:从语音到推销

    项目背景 EnCode 2025 项目的目标是创建一个高质量、语音自然流畅的AI销售代理,实现与真人近乎无延迟的交互体验。 为此,我构建了一个系统,能够完整处理在线辅导机构的销售对话流程——从问候潜在客户到了解需求并推荐课程。整个过程都以积极、人性化的语气进行,如同一位不知疲倦、状态始终在线的销售人…

    2025年12月13日
    000
  • 我的 Python 之旅:从禅宗到列表

    开启我的编程学习日志!我将记录我的编码学习历程,分享项目成果,并与大家一起成长。 为何踏上这段旅程? 我是一名全职数据分析师,也是三个孩子的父亲。日常工作中大量使用定性和定量分析方法,为了提升技能,我决定重拾编码热情,学习 Python 正是将专业知识与新技能结合的绝佳途径。 这个博客记录了我学习过…

    2025年12月13日
    000
  • Python 中的数据结构 -Stack

    Python 中的堆栈,如同其他编程语言一样,是一种遵循后进先出 (LIFO) 原则的线性数据结构。这意味着最后添加的元素将最先被移除。 堆栈的应用场景: 想象一下一堆盘子,你只能添加或移除最上面的盘子。 常见的堆栈操作包括“push”(压入,添加元素)、“pop”(弹出,移除顶部元素)和“peek…

    2025年12月13日
    000
  • Python 库在数据科学中的威力

    python数据科学:核心库指南 Python凭借其丰富的库生态系统在数据科学领域占据主导地位,这些库涵盖了数据分析管道的各个阶段。从数据处理到可视化、机器学习和深度学习,Python库提供了强大的工具来应对各种挑战。本指南深入探讨数据科学中最重要的Python库,阐述其特性、功能和实际应用。 1.…

    2025年12月13日
    000
  • 探索人工智能工具的世界:彻底改变工作和学习

    ChatGPT:对话式AI助手 简介:ChatGPT是OpenAI打造的先进对话式AI,擅长理解和生成类人文本。它非常适合头脑风暴、撰写邮件、辅助编程和学习。 应用场景: 内容和文章创作代码片段编写和调试各种主题问题的解答 链接:chatgpt GitHub Copilot:编程伙伴 简介:由Ope…

    2025年12月13日
    000
  • 将 Mac OSX 图书亮点导出到 Obsidian Vault 或 Markdown 文件

    readwise 功能强大,但对于跨平台管理笔记和高亮的用户而言,其优势更明显。我主要用于电子书高亮,而使用 readwise 的主要目的就是将这些高亮和笔记导入到 obsidian 中。我习惯在网络上做笔记,使用 obsidian web clipper,甚至在 ipad 上,自从发现 orion…

    2025年12月13日 好文分享
    000

发表回复

登录后才能评论
关注微信