上一篇文章已经对Swift的面向协议编程做了介绍,接下来该篇文章将使用面向协议开发(POP)来做下实际的应用
在实际开发中,自定义View基本上是必须的,相信这对我们来说都是比较简单,不过我们还是来回顾一下下~
面向对象开发
1 新建一个UIView的FirstTypeView
2 创建一个View的xib文件
3 设置xib对应的class进行绑定4 在FirstTypeView.swift 中实现一个类方法,方便我们外部用xib来初始化FirstTypeView
import UIKitclass FirstTypeView: UIView {}extension FirstTypeView { class func loadFromNib() -> FirstTypeView { return Bundle.main.loadNibNamed("\(self)", owner: nil, options: nil)?.first as! FirstTypeView }}复制代码
在外部只要调用FirstTypeView的loadFromNib方法就可以初始化一个View来使用了。好,现在又有一个类SecondTypeView,也是要求使用xib来初始化view。这时我们就会想,一样的加载xib的方法,那我们就把它抽取出来放到父类就可以了。这里的父类以BaseView.swift为例
父类的主要实现代码extension BaseView { class func loadFromNib() -> BaseView { return Bundle.main.loadNibNamed("\(self)", owner: nil, options: nil)?.first as! BaseView }}复制代码
那我们的FirstTypeView和SecondTypeView只需要直接继承于BaseView就可以了,在其它地方初始化view也很方便
let firstView = FirstTypeView.loadFromNib()view.addSubview(firstView)let secondView = SecondTypeView.loadFromNib()view.addSubview(secondView)复制代码
好,现在FirstTypeView里声明了一个属性name,SecondTypeView声明的属性为age,假如我们现在要使用各自对应的属性,这时是直接点不出来的,需要先进行强转
let firstView = FirstTypeView.loadFromNib() as! FirstTypeViewfirstView.name = "LXF"view.addSubview(firstView)let secondView = SecondTypeView.loadFromNib() as! SecondTypeViewsecondView.age = 100view.addSubview(secondView)复制代码
但是这样觉得不是很方便,还需要进行强转,我们能不能在BaseView里面搞定它呢?如果是Swift 2.x 的话是可以的
extension BaseView { class func loadFromNib() -> Self { // 注意这里是大写的S return Bundle.main.loadNibNamed("\(self)", owner: nil, options: nil)?.first as! Self }}复制代码
但是现在Swift 3.0已经不支持这种写法了,会报错。这个时候如果使用面向协议的开发就很方便了。
面向协议开发
将BaseView删除,FirstTypeView和SecondTypeView改回继承于UIView
1 新建一个Swift文件 Nibloadable.swift
2 实现协议方法
协议中不允许定义类方法,需改为静态方法
import UIKitprotocol Nibloadable {}extension Nibloadable { static func loadFromNib() -> Self { return Bundle.main.loadNibNamed("\(self)", owner: nil, options: nil)?.first as! Self }}复制代码
3 遵守协议
class SecondTypeView: UIView, Nibloadable { var age: Int = 10}复制代码
这样就可以了,而且你在调用loadFromNib方法时可以发现,类型是对应上的
好了,面向协议开发的应用就记录到这里,希望能帮助到大家!