面向协议编程
在使用 Swift 时,可以为结构体和枚举类型声明方法,其使用起来更加方便,对于一些简单模型,或者对象的组合模型,使用结构体更为简便。
对于面向对象编程(OOP,Object Oriented Programming)而言,类能够更好的描述模型,并且可以将抽象出的通用属定义在父类中。
但是,不管是 Objective-C 还是 Swift 语言,都不支持多继承,但是他们都可以遵循多个协议。实际编程时,可以将不同的特性分类,定义在不同的协议中,然后通过遵循协议的方式来实现一类特性。
在 Objective-C 中,值类型既不继承其他类型,也不能实现协议。如果要更多的使用值类型,那么抽象出通用的属性给不同值类型继承,就十分必要。而 Swift 做得更多,所有的类型都可以遵循多个协议,而且协议本身可以实现默认的方法,如此,针对一类操作定义一个协议,那么只要遵循该协议,就可以获得相应操作。
如同一个孩子的发质继承自父亲,而眼睛更像母亲,那么如果使用面向对象的方式进行描述,无法支持多继承的情况下,只能冗余的将特性既声明在父类中又声明在子类中。而如果将特性声明在协议中,父类和子类遵循相同的协议就表示拥有同一个特质,而子类再遵循一个描述母亲眼睛特性的协议即可。
所以,面向协议编程(POP,Protocol Oriented Programming)是通过不同协议的继承和组合来描述事物。可以说,面向对象更多的是对一类事物通用特性的抽象,而面向协议更多的是对不同事物通用特性的抽象。前者更注重整体特性的抽象,后者是对特性更细粒度的抽象。通过不同特性的组合来描述一类事物,比通过层层继承来描述一类事物更加简化,更加具体,代码冗余度也会下降。
例如,在应用架构的设计时,也可以使用面向协议编程的思想,我们可以将以前写入基类中的一些属性或方法,定义在协议中,通过遵循协议来使相应的类型获得需要的属性或方法。
protocol HXJRootProtocol: NSObject {
var isSafeContentViewEnable: Bool { get set }
var safeContentView: UIView! { get }
}
extension UIViewController: HXJRootProtocol {
private static let safeContentViewTag: Int = 1000
var safeContentView: UIView! {
assert(isSafeContentViewEnable, "需要先设置 isSafeContentViewEnable 为 true 才可以使用 safeContentView")
if let view = view.viewWithTag(UIViewController.safeContentViewTag) {
return view
}
return nil
}
var isSafeContentViewEnable: Bool {
set {
if newValue {
if nil != view.viewWithTag(UIViewController.safeContentViewTag) { return }
addSafeContentView()
}else {
if let view = view.viewWithTag(UIViewController.safeContentViewTag) {
view.removeFromSuperview()
}
}
}
get {
return view.viewWithTag(UIViewController.safeContentViewTag) != nil
}
}
private func addSafeContentView() {
let safeContentView = UIView.init()
view.addSubview(safeContentView)
safeContentView.backgroundColor = .clear
safeContentView.tag = UIViewController.safeContentViewTag
safeContentView.translatesAutoresizingMaskIntoConstraints = false
safeContentView.leftAnchor.constraint(equalTo: self.view.leftAnchor).isActive = true
safeContentView.rightAnchor.constraint(equalTo: self.view.rightAnchor).isActive = true
if #available(iOS 11.0, *) {
safeContentView.topAnchor.constraint(equalTo: self.view.safeAreaLayoutGuide.topAnchor).isActive = true
safeContentView.bottomAnchor.constraint(equalTo: self.view.safeAreaLayoutGuide.bottomAnchor).isActive = true
} else {
safeContentView.topAnchor.constraint(equalTo: self.topLayoutGuide.bottomAnchor).isActive = true
safeContentView.bottomAnchor.constraint(equalTo: self.bottomLayoutGuide.topAnchor).isActive = true
}
}
}
在上面的代码中,通过使 UIViewController 遵循 HXJRootProtocol 协议,来添加一个安全区域的视图。
虽然无法为一个类型扩展存储属性,但是如果有必要,可以使用运行时的方法来实现该需求。