原文1:https://www.raywenderlich.com/86477/introducing-ios-design-patterns-in-swift-part-1
原文2:http://www.raywenderlich.com/90773/introducing-ios-design-patterns-in-swift-part-2
中文翻译:https://swift-design-patterns.books.yourtion.com/Chapter01/iOS-Design-Patterns.html
以下为个人针对这两篇文章的理解与总结
设计模式
一个好的设计模式可以使解决方案变成可复用且利于理解的,并且降低代码的耦合度
Cocoa中常见的设计模式
Cocoa中常见的几种设计模式:
- 创建型 (Creational):单例模式 (Singleton)
- 结构型 (Structural):MVC、装饰者模式 (Decorator)、适配器模式 (Adapter)、外观模式 (Facade)
- 行为型 (Behavioral):观察者模式 (Observer)、备忘录模式 (Memento)
MVC
Model-View-Controller 是 Cocoa 的一部分,最常用的设计模式之一。MVC中对象按职责被分类。
- 模型层 (Model) :存储数据并且定义如何操作这些数
- 视图层 (View) :负责模型层的可视化展示,并且负责用户的交互,一般来说都是继承自 UIView
- 控制器 (Controller) :控制器是整个系统的掌控者,它连接了模型层和数据层,并且把数据在视图层展示出来,监听各种事件,负责数据的各种操作。
MVC 的强大之处:三大元素,各司其职,减少依赖。
单例模式 - Singleton
确保每个指定的类只存在一个实例对象,并且可以全局访问那个实例。一般情况下会使用延时加载的策略,只在第一次需要使用的时候初始化。
iOS 中的单例模式:NSUserDefaults.standardUserDefaults()
UIApplication.sharedApplication()
UIScreen.mainScreen()
NSFileManager.defaultManager()
使用场景:某些情况实例只需要一份,如UIScreen.mainScreen()
外观模式 - Facade
外观模式将复杂的业务系统包装为简单的接口,只将接口暴露给使用者。如果直接把业务的所有接口直接暴露给使用者,使用者需要单独面对这一大堆复杂的接口,学习成本很高,而且存在误用的隐患。
使用场景:当我们有大量的类并且很复杂而且也很难理解的时候,外观模式可以很理想的解决问题。
作用:外观模式把使用和背后的实现逻辑成功解耦,同时也降低了外部代码对内部工作的依赖程度。如果底层的类发生了改变,外观的接口并不需要做修改。
装饰者模式 - Decorator
装饰者模式可以动态的给指定的类添加一些行为和职责,不对原代码进行修改。
iOS中的实现方式:扩展 (Extension) 和委托 (Delegation)
拓展
扩展可以在不继承的情况下,给已存在的类、结构体或者枚举类添加一些新的功能。尤其可以在没有访问权限的情况下扩展已有类。意味着可以扩展 Cocoa 类。
委托
委托机制下,一个对象(委托者)可以和另一个对象(被委托者)相关联,委托者通知被委托者完成委托事项。用协议可以实现。
适配器模式 - Adapter
适配器把自己封装起来然后暴露统一的接口给其他类,其他类就得适配这个接口,这样即使其他类的接口各不相同,也能相安无事,一起工作。
iOS中的实现:苹果通过委托实现了适配器模式。比如,如果一个类遵循了 NSCoying 的协议,那么它一定要实现 copy 方法。
观察者模式 - Observer
在观察者模式里,一个对象在状态变化的时候会通知另一个对象。参与者并不需要知道其他对象的具体是干什么的,这是一种降低耦合度的常用于在某个属性改变的时候通知关注该属性的对象的设计模式。
iOS中使用场景:观察者注册监听,然后再状态改变的时候,所有观察者们都会收到通知。如在 MVC 里,观察者模式使得 Model 对象和 View 对象进行交流,但并不是通过关联。Model变化时,发送消息通知Controller,Controller更新View。
iOS中的实现:Cocoa使用两种方式实现了观察者模式: Notification 和 Key-Value Observing (KVO)。
通知 - Notification
一个对象 (发布者) 向其他对象 (订阅者) 发送消息。发布者永远不需要知道订阅者的任何数据。
iOS中使用场景:Apple 对于通知的使用很频繁,比如当键盘弹出或者收起的时候,系统会发送 UIKeyboardWillShowNotification/UIKeyboardWillHideNotification
,应用切到后台的时候,会发送 UIApplicationDidEnterBackgroundNotification
。UIApplication.swift
中有很多系统通知处理函数。
键值观察 - KVO
在 KVO 里,对象可以注册监听任何属性的变化,不管它是否持有。苹果 KVO 编程指南。
示例代码:
coverImage.addObserver(self, forKeyPath: "image", options: NSKeyValueObservingOptions([.New, .Old]), context: nil)
注:image是coverImage对象的属性。
备忘录模式 - Memento
备忘录模式实现对象状态的存储。即将对象存在了某个地方,然后在以后的某个时间再把它恢复出来,而不会打破它本身的封装性,私有数据依旧是私有数据。
iOS中的实现:苹果通过归档(Archiving)的方法来实现备忘录模式,一个类需要遵守 NSCoding 协议,才能成为可被归档的。归档将对象转化成了流然后在不暴露内部属性的情况下存储数据。苹果的归档和序列化文档。