[IOS架构]Swinject 依赖注入框架

在本文中,我将介绍依赖注入的基础知识,以及如何使用Swinject框架将依赖注入应用到iOS项目中。

什么是依赖

依赖是我们代码中两个模块之间的耦合(在面向对象语言中,指的是两个类),通常是其中一个模块使用另外一个提供的功能。

依赖有什么不好?

从上层到底层依赖都是危险的,因为我们在某种程度上把两个模块进行耦合,这样当需要修改其中一个模块时,我们必须修改与其耦合的模块的代码。这对于创建一个可测试的app来说是很不利的,因为单元测试要求测试一个模块时,要保证它和app中其他模块是隔离的。

举个例子

class Module1{
   var module2:Module2 

   init (){
      module2 = Module2()
   }

   func doSomething(){
      ...
      module2.doSomethingElse();
      ...
   } 
}

如何在不测试doSomethingElse函数的前提下测试doSomething函数呢?如果测试失败,是哪个函数导致的呢?我们不得而知。如果doSomethingElse函数在数据库中保存数据或者向服务器端发起API请求,那么事情将变得更加糟糕。

每当敲下new关键字我们都应该意识到这可能是需要避免的强依赖。编写更少的模块也不是解决方案,不要忘记单一职责原则。

依赖反转

如果不能在一个模块内部初始化另外的模块,那么需要以其他的形式初始化这些模块。你能想象如何实现吗?没错,通过构造函数。这基本上就是依赖反转原则的涵义了。你不应该依赖具体的模块对象,应该依赖抽象。

前面的代码应该修改为:

 class Module1{
   var module2:Module2 

   init(module2: Module2){
      self.module2 = module2
   }

   func doSomething(){
      ...
      module2.doSomethingElse();
      ...
   } 
}

什么是依赖注入呢?

依赖注入(Dependency Injection, DI)是一种技术,在这种技术中,可以从实体自身的范围之外设置实体的依赖项,将整个系统转换为松散耦合的模块。想象一下,我们可以提供一个模块,这个模块包含了对应用程序其他组件的引用,这样你就可以避免UIViewController之间的通信模式。

从上面的例子我们知道,通过构造函数传递依赖(注入),从而把创建模块的任务从另一个模块内部抽离出来。对象在其他地方创建,并以构造函数参数的形式传递给另一个对象。

但新问题出现了。如果我们不能在模块内部创建其他的模块,那么必须有个地方对这些模块进行初始化。另外,如果我们需要创建的模块的构造函数包含大量的依赖参数,代码将变得丑陋和难以阅读,app中将存在大量传递的对象。依赖注入正是为解决这类问题而诞生的。

我们需要在app中提供另一个模块,专门负责提供其他模块的实例并注入他们的依赖,这个模块就是依赖注入器,模块的创建集中于app中的一个统一的入口。

还是举个例子来说明问题吧。

首先,如果没有进行依赖注入的情况

第一步,创建Cat类:

class Cat {
    let name: String

    init(name: String) {
        self.name = name
    }

    func sound() -> String {
        return "Miao !"
    }
}

第二步,创建Person类:

class Person {
    let pet = Cat(name: "nimo")

    func play() -> String {
        return "I'm playing with \(pet.name). \(pet.sound())"
    }
}

Person类中关联了Cat类属性。

实际使用:

let per = Person()
print(per.play())

输出:

// 输出 "I'm playing with nimo. Miao!"

问题来了,如果我不想养猫,我想养狗了,那我是不是就得新建一个Person2,关联一个Dog类呢?
所以必须要对两个类的依赖进行解耦, 并且改变为依赖抽象,这样之后再进行依赖替换的时候就很容易了。

其次,我们来尝试解耦

第一步,我们先将宠物抽象成一个接口协议,使用者不用具体实现,只是依赖这个协议即可:

protocol AnimalType {
    var name: String { get }
    func sound() -> String
}

第二步,让 Cat 类实现这个协议:

class Cat: AnimalType {
    let name: String

    init(name: String) {
        self.name = name
    }

    func sound() -> String {
        return "Miao!"
    }
}

第三步,Person中关联一个依赖于AnimalType的pet,而非Cat的具体实现,并构造一个初始化方法,将pet传入:

class Person {
    let pet: AnimalType

    init(pet: AnimalType) {
        self.pet = pet
    }

    func play() -> String {
        return "I'm playing with \(pet.name). \(pet.sound())"
    }
}

具体使用:

let catPerson = Person(pet: Cat(name: "nimo"))
print(catPerson.play()) // 输出 "I'm playing with nimo. Miao!"

如果换成养狗,则创建个Dog:

class Dog: AnimalType {
    let name: String

    init(name: String) {
        self.name = name
    }

    func sound() -> String {
        return "wwww!"
    }
}

具体使用:

let dogPerson = Person(pet: Dog(name: "hah"))
print(dogPerson.play()) // 输出 "I'm playing with hah. wwww!"

以上是通过抽象出一个接口协议,代替了具体实现,如果项目中依赖关系多且复杂,使用Swinject进行依赖注入就比较方便了。

最后,我们使用Swinject来进行依赖注入

Swinject是一个很棒的用于Swift项目的DI框架,而且它是开源的。它使用泛型以一种非常简单的方式解耦你的代码。

第一步,通过CocoaPods将其添加到您的项目中:

pod ‘Swinject’

导入:

import Swinject

第二步,包装container,放到一个统一的类中:

import Swinject
class DIContainer {
    static let container:Container = {
        let con = Container()
        con.register(AnimalType.self) { _ in  Cat(name: "Nimo") }
        con.register(Person.self) { r in
           Person(pet: r.resolve(AnimalType.self)!)
        }
        return con
    }()
}

第三步,具体使用:

    let container = DIContainer.container
    override func viewDidLoad() {
        super.viewDidLoad()
        
        let per = container.resolve(Person.self)!
        print(per.play())
    }

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,393评论 5 467
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,790评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,391评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,703评论 1 270
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,613评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,003评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,507评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,158评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,300评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,256评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,274评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,984评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,569评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,662评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,899评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,268评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,840评论 2 339

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,076评论 1 32
  • 第3章 基本概念 3.1 语法 3.2 关键字和保留字 3.3 变量 3.4 数据类型 5种简单数据类型:Unde...
    RickCole阅读 5,092评论 0 21
  • Dagger2是个什么东西呢?依赖注入,这是个啥玩意?嗯,在学这个东西的时候我们得了解一些知识点: 知识点呀 依赖...
    SHERLOCKvv阅读 1,187评论 0 6
  • 纳兰性德的《木兰花·拟古决绝词柬友》中这两句“人生若只如初见,何事秋风悲画扇”不知为多少人所传唱。本是凄美的爱...
    紫楼兰阅读 419评论 8 1
  • 不要去改变老人的格局 比如东西摆放的位置
    茄子要盐水阅读 140评论 0 0