Swift优雅的使用Protocol(一) 从OC的代理说起

2018-12-14  本文已影响51人  seasonZhu

因为涉及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.png

weak这个关键词只能修饰类(class),然而Protocol不管是类(class)还是结构体(struct)或者说枚举(enum)(其实枚举也是结构体的一种)都可以遵守的.

所以如果如上图中设置代理属性,并用weak修饰,那么协议要么就继承NSObjectProtocol,或者用: class修饰,说明这个协议只在类中使用.

整体而言,不管是可选协议的实现方法用@objc修饰协议然后又使用@objc optional修饰方法,这样的写法都很不Swift.

使用 Protocol的Extension可以很好的解决问题.

而weak修饰导致必须继承NSObjectProtocol,我个人更加喜欢用: class的方式来解决.

最后你会发现,在这篇文章中对于Protocol,我都是说某个类(class)遵守某个协议(protocol).

遵守这个词可不可以换成继承呢? ------继承某个协议(protocol)?

好了,那就是下篇我会继续说明的问题.如果你记得我在Swift优雅的使用Extension最后的一点引子,你也许就对继承某个协议(protocol)不那么陌生了.

上一篇 下一篇

猜你喜欢

热点阅读