因为涉及Swift中Protocol的知识点会非常的多,所以我将这部分拆分为4部分来进行编写,我会从OC时代的Protocol在Swift中的运用开始,然后逐渐深入的到Swift的Protocol中,进而对面向协议编程思想,泛型等高级用法.
本篇只是对于OC的代理模式在Swift中的运用,如果你对这个以及非常了解,那么可以不看这篇文章
那么好,正文开始,说起OC中的代理模式,我们一般用于在参数回调中,其实随着Block的深入运用,使用Protocol感觉非常的繁琐起来.
这里我们不过多的讨论什么时候用代理,什么时候用Block,我们重点说一说,代理在Swift中使用的注意事项.
我们先看一个非常经典的例子,UITableView和DataSource,这一个例子就足够说明问题.
public protocol UITableViewDataSource : NSObjectProtocol {
@available(iOS 2.0, *)
public func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int
// Row display. Implementers should *always* try to reuse cells by setting each cell's reuseIdentifier and querying for available reusable cells with dequeueReusableCellWithIdentifier:
// Cell gets various attributes set automatically based on table (separators) and data source (accessory views, editing controls)
@available(iOS 2.0, *)
public func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell
@available(iOS 2.0, *)
optional public func numberOfSections(in tableView: UITableView) -> Int // Default is 1 if not implemented
@available(iOS 2.0, *)
optional public func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String? // fixed font style. use custom view (UILabel) if you want something different
@available(iOS 2.0, *)
optional public func tableView(_ tableView: UITableView, titleForFooterInSection section: Int) -> String?
.
.
.
我们都知道, 数据源中numberOfRowsInSection和cellForRowAt indexPath两个方法是必须要是实现,而其他的数据源方法都是可选实现.
注意看到optional这个关键词没有吗?没错,这点也很OC相似,使用optional修饰的方法可以进行可选实现.
但是协议方法中能够使用optional修饰,需要满足这样几种约定的写法,不要问为什么,遵守规则即可,不然编译不过.
image.png定义协议前,请使用@objc修饰protocol关键字
在可选的方法前加上@objc optional修饰
至于遵守这个协议的类,是继承NSObject或者不继承,都没关系
但是如果你的协议是继承于NSObjectProtocol,那么类必须继承NSObject!
至于为啥UITableViewDataSource可以不用这么麻烦的修饰,我只能说系统定义,那是苹果爸爸干的事.
那么我说一下另外一个点,就是注意看
protocol UITableViewDataSource : NSObjectProtocol
tableView的代理继承自NSObjectProtocol协议,这个是一个非常重要的点,Swift中协议是可以继承的.
如果你仔细看看NSObjectProtocol这个协议,你最终会追根溯源到NSObject,NSObject这个类在Swift中是遵守了NSObjectProtocol这个协议的.
也就是说Cocoa框架下衍生出来的类都遵守了NSObjectProtocol这个协议.
然后,我要说代理的使用,我们一般会在类中用weak修饰这个代理变量,在Swift中这个代理变量我们用可选类型进行修饰.
我们看看下面这个截图,你会发现,delegate1报错了.
image.pngweak这个关键词只能修饰类(class),然而Protocol不管是类(class)还是结构体(struct)或者说枚举(enum)(其实枚举也是结构体的一种)都可以遵守的.
所以如果如上图中设置代理属性,并用weak修饰,那么协议要么就继承NSObjectProtocol,或者用: class修饰,说明这个协议只在类中使用.
整体而言,不管是可选协议的实现方法用@objc修饰协议然后又使用@objc optional修饰方法,这样的写法都很不Swift.
使用 Protocol的Extension可以很好的解决问题.
而weak修饰导致必须继承NSObjectProtocol,我个人更加喜欢用: class的方式来解决.
最后你会发现,在这篇文章中对于Protocol,我都是说某个类(class)遵守某个协议(protocol).
遵守这个词可不可以换成继承呢? ------继承某个协议(protocol)?
好了,那就是下篇我会继续说明的问题.如果你记得我在Swift优雅的使用Extension最后的一点引子,你也许就对继承某个协议(protocol)不那么陌生了.
网友评论