Swift 可选接口

Objective-C 中的 Protocol 里存在 @optional 关键字,被这个关键字修饰的方法并非必须要被实现。比如最常见的 UITableViewDataSource 和 UITableViewDelegate。但是在 Swift 中的 Protocol 的所有方法都是必须被实现的。那么我们怎么样才能在 Swift 实现可选接口呢?

  • @objc & optional
  • 接口扩展

@objc & optional

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
@objc protocol MyProtocol {
func doSomething()

@objc optional func doSomething_optional()
}

class MyClass: NSObject, MyProtocol {
func doSomething() {

}
}

/// 编译不通过
/// Non-class type 'MyStruct' cannot conform to class protocol 'MyProtocol'
struct MyStruct: MyProtocol {

}

原生的 Swift Protocol 里是没有可选项的,所有定义的方法都是必须被实现的。如果我们需要像 Objective-C 里那样定义可选的方法,可以直接将接口本身定义为 Objective-C 的。

优势

  • 直接以 Objective-C 的接口使用方式来使用。

劣势

  • 使用 @objc 修饰的 Protocol 就只能被 class 实现。对于 struct 和 enum 类型,我们无法实现此种方式定义的接口。

接口扩展

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
extension MyProtocol {
func doSomething_optional() {
print("Default implementing by extension protocol.")
}
}

class MyClass: MyProtocol {
func doSomething() {
// do something
}
}

struct MyStruct: MyProtocol {
func doSomething() {
// do something
}
}

enum MyEnum: MyProtocol {
func doSomething() {
// do something
}
}

虽然 Swift 中定义的 Protocol 的方法都是必须实现的,但是我们可以利用 protocol extension 的方式给出部分方法的默认实现。

优势

  • 利用 extension 给出默认实现的接口可以供 class、struct 及 enum 使用。

劣势

  • 对于有返回值的接口,默认实现总是需要提供一个合适的返回值,但是并不总是有合适的默认值返回。

例如:

1
2
3
protocol MyProtocol {
func calculateRadian() -> Double
}

这里的 Protocol 定义的方法返回一个计算出来的弧度,但是这时我们就很难找到一个合适的默认值去默认实现接口的方法。


参考

Swifter 100 个 SWift 必备 Tips