设计模式之依赖倒置原则

相关链接:
0. 设计模式之六大原则总结
1. 设计模式之单一职责原则
2. 设计模式之里式替换原则
3. 设计模式之依赖倒置原则
4. 设计模式之接口隔离原则
5. 设计模式之迪米特法则
6. 设计模式之开闭原则

1.1 定义
  • 高层模块不应该依赖低层模块,二者都应该依赖其抽象;
  • 抽象不应该依赖细节;
  • 细节应该依赖抽象
1.2 问题由来

类 A 直接依赖类 B,假如要将类 A 改为依赖类 C ,则必须通过修改类 A 的代码来达成。这种场景下,类 A一般是高层模块,负责复杂的业务逻辑;类 B 和类 C 是低层模块,负责基本的原子操作;假如修改类 A,会给程序带来不必要的风险。

1.3 解决方案

将类 A 修改为依赖接口 Interface1,类 B 和类 C 各自实现接口 Interface2,类 A 通过接口 Interface1 间接与类 B 或者类 C 发生联系,则会大大降低修改类 A 的几率。

1.4 具体分析

依赖倒置原则基于这样一个事实:相对于细节的多变性,抽象的东西要稳定的多。以抽象为基础搭建起来的架构比以细节为基础搭建起来的架构要稳定的多。抽象指的是接口或者抽象类,细节就是具体的实现类,(iOS 中可以理解为抽象就是协议,细节是实现该协议的实现类),使用接口或者抽象类的目的是制定好规范和契约,而不去涉及任何具体的操作,把展现细节的任务交给他们的实现类去完成。

依赖倒置原则的核心思想是面向接口编程,达到解耦的过程。

1.5 举例说明

我们用一个例子来说明面向接口编程相对于面向实现编程好在什么地方。母亲给孩子讲故事,只要给她一本书,她就可以照着书给孩子讲故事了,代码如下:

class Book : NSObject{
    func getContent() -> String {
        return "在很久很久之前....";
    }
}

class Mother : NSObject {
    func narrate(book : Book) {
        print("妈妈开始将故事");
        print(book.getContent());
    }
}

// 执行的代码
let mother = Mother()
mother.narrate(book: Book())

运行结果:

妈妈开始讲故事
很久很久以前有一个阿拉伯的故事……
1.6 举例进阶

假如有一天,孩子长大了,需要了解国家大事了,不是看书,而是看报纸了,让这位母亲讲一下报纸的新闻,报纸的代码如下:

class Newspaper : NSObject{
    func getContent() -> String {
        return "2020 年的春运已经开始了....";
    }
}

但是,目前来说,这位母亲办不到啊,因为 Mother 类里面只有一个narrate方法,参数是Book,这时,还得需要母亲学习如何读报纸(修改 Mother 类的代码,添加方法),需要修改 Mother 才能读,但是实际上都是文字啊,Mother 再去学习,学啥啊?
假如以后需求变成杂志、网页呢?还得需要不断的修改 Mother,这显然不是好的设计。原因就是 Mohter与 Book 之间的耦合性太高了,必须降低他们之间的耦合度才行。

我们以 iOS 为例,定义一个协议:读物,只要是可以读的都属于读物:

protocol IReader {
    // 获取读物内容
    func getContent() -> String;
}

Mother类和接口 IReader 之间发生依赖关系,而 Book 和 Newspaper 都属于读物的范畴,他们各自都去实现 IReader 接口,这样就符合依赖倒置原则了,代码修改为:

class Book : NSObject, IReader{
    func getContent() -> String {
        return "在很久很久之前....";
    }
}

class Newspaper : NSObject, IReader{
    func getContent() -> String {
        return "2020 年的春运已经开始了....";
    }
}

class Mother : NSObject {
    func narrate(reader : IReader) {
        print("妈妈开始将故事");
        print(reader.getContent());
    }
}
1.7 举例总结

这样修改之后,无论以后想去看网页、杂志,都不需要再修改 Mother 类了。这只是一个简单的例子,实际情况中,代表高层模块的 Mother 类将负责完成主要的业务逻辑,一旦需要对它进行修改,引入错误的风险极大。所以遵循依赖倒置原则可以降低类之前的耦合性,提高系统的稳定性,降低修改程序造成的风险。

1.8 总结

采用依赖倒置原则给多人并行开发带来了极大的便利,比如上例中,原本 Mother 类与 Book 类直接耦合时,Mother 类必须等 Book 类编码完成后才可以进行编码,因为 Mother 类依赖于 Book 类,修改后的程序则可以同时开工,互不影响,因为 Mother 与 Book 类之间一点关系也没有。参与协作开发的人越多、项目越庞大,采用依赖倒置原则的意义就越重大。

在实际编程时,我们一般需要做到如下 3 点:

  • 低层模块尽量都要有抽象类或接口,或者两者都有
  • 变量的声明类型尽量是抽象类或接口
  • 使用继承时遵循里式替换原则

依赖倒置原则的核心就是要我们面向接口编程,理解了面向接口编程,也就理解了依赖倒置。

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

推荐阅读更多精彩内容