依赖反转和依赖注入

依赖反转和依赖注入是软件开发中两个密切相关的概念, 由于它们的重叠方面经常被混淆. 虽然它们可以协同工作, 但在实现结构良好的应用方面却有着不同的目的. 下面通过一个示例来分析它们的区别:

依赖反转:

  • 原则: 高层模块不应依赖于低层模块; 两者都应依赖于抽象模块.
  • 重点: 架构和模块之间的关系.
  • 目标*: 松耦合, 提高灵活性.

设想一个 Android 应用要处理不同类型的网络连接(WiFi, 蜂窝).

kotlin 复制代码
Copy// Abstraction (high-level)
interface NetworkInterface {
    fun connect(): Boolean
    fun disconnect(): Boolean
}

// Concrete implementations (low-level)
class WiFiNetwork : NetworkInterface {
    override fun connect(): Boolean {
        // Connect to WiFi network
    }

    override fun disconnect(): Boolean {
        // Disconnect from WiFi network
    }
}

class CellularNetwork : NetworkInterface {
    override fun connect(): Boolean {
        // Connect to cellular network
    }

    override fun disconnect(): Boolean {
        // Disconnect from cellular network
    }
}

// High-level module (depends on abstraction)
class NetworkManager(private val networkInterface: NetworkInterface) {
    fun manageNetwork() {
        if (networkInterface.connect()) {
            // Perform network operations
        } else {
            // Handle connection failure
        }
    }
}

让我们在 NetworkManager 的帮助下分解依赖反转

  • **低级: ** 网络连接的具体实现(WiFiNetwork, CellularNetwork).
  • **高级: ** 一个处理整个网络通信的NetworkManager类.
  • **不反转: ** NetworkManager 直接依赖于 WiFiNetworkCellularNetwork 类, 使其与特定实现紧密耦合.
  • **反转: ** NetworkManager 依赖于抽象的 NetworkInterface 接口, 该接口定义了基本的网络操作. WiFiNetworkCellularNetwork都实现了NetworkInterface. 这使 NetworkManager 与具体实现分离, 从而可以在它们之间轻松切换或添加新的实现, 而无需修改核心逻辑.

NetworkManager依赖于抽象的NetworkInterface, 而不是具体的实现, 从而实现了依赖反转.

现在让我们深入了解依赖注入.

依赖注入:

  • 原则: 从外部向类提供依赖, 而不是在内部创建依赖.
  • 重点: 在运行时实现依赖反转原则.
  • 目标: 提高可测试性, 模块化和配置灵活性.

继续上述场景, 让我们在运行时将特定的 NetworkInterface 实现注入到 NetworkManager 中.

在运行时注入特定的 NetworkInterface 实现, 展示了依赖注入, 这样就可以在不修改 NetworkManager 的情况下切换实现.

kotlin 复制代码
Copy// Injecting a specific implementation at runtime
fun chooseNetwork(isWiFiAvailable: Boolean): NetworkInterface {
    return if (isWiFiAvailable) {
        WiFiNetwork()
    } else {
        CellularNetwork()
    }
}

val networkManager = NetworkManager(chooseNetwork(true)) // Inject WiFiNetwork
  • **注入方式: ** 使用构造器注入, 在创建 NetworkManager 对象时传递所需的 NetworkInterface 实例(WiFiNetworkCellularNetwork).
  • 优点 : 你可以根据上下文(如可用的 WiFi)选择网络实现, 使代码更具适应性. 通过模拟不同的 NetworkInterface 实现, 测试变得更容易.

摘要:

  • 依赖反转: 定义了整体架构以及模块之间的依赖关系.
  • 依赖注入: 是一种通过从依赖类外部提供依赖关系来实现反转原则的技术.

在安卓系统中, Dagger/HiltKoin等库可以通过注入来帮助管理依赖关系, 从而更容易在应用中实现依赖反转原则.

请记住, 这两个概念共同作用于创建松散耦合, 灵活且结构良好的应用. 了解它们的区别将有助于你在 Android 项目中设计和实现高效的代码.

我希望这篇结合 Kotlin Android 示例的详细解释能澄清依赖反转和依赖注入之间的区别.

Happy Coding!

相关推荐
BreezeDove9 小时前
【Android】AS项目自动连接mumu模拟器配置
android
乐世东方客12 小时前
备份脚本记录(binlog文件+mysql+mongo)
android·数据库·mysql
私人珍藏库12 小时前
[Android] 视频下载鸟 v20.02 会员
android·人工智能·智能手机·app·工具·多功能
zh_xuan13 小时前
tv浏览网页工具
android·tv浏览网页
Carson带你学Android14 小时前
Compose 终于上线 FlexBox:换行与弹性伸缩 都轻松搞定!
android·composer
私人珍藏库14 小时前
[Android] 三维山水全景地图-3D地形全景观测地图
android·3d·app·工具·软件·多功能
dengyuezhe806015 小时前
《C++ 异常机制与智能指针:从原理到实现》
android·java·c++
Wonderful U15 小时前
Python+Django实战|企业办公用品申领管理系统:物资入库、库存预警、申领审批、归还登记、损耗统计、供应商对账
android·python·django
plainGeekDev15 小时前
网络状态监听 → ConnectivityManager + Flow
android·java·kotlin
楠目15 小时前
CVE-2013-4547 Nginx URI解析漏洞利用总结
android