标签:
和我一起参加9 月 1 日 - 9月 2 日在纽约举办的 Swift 社区庆典??吧!使用优惠码 NATASHATHEROBOT 可以获得 $100 的折扣!
我最近做了个 Swift 面向协议编程实践(POP??) 的演讲。视频还在处理中。另一方面,这是演讲中 POP 视图部分的文本记录,供我和其他任何人作参考!
假设我们要做一款展示全球美食图片和信息的 App。这需要从 API 上拉取数据,那么,用一个对象来做网络请求也就是理所当然的了:
struct FoodService {
|
一旦我们创建了异步请求,就不能使用 Swift 內建的错误处理来同时返回成功响应和请求错误了。不过,倒是给练习 Result 枚举创造了机会(更多关于 Result 枚举的信息可以参考 Error Handling in Swift: Might and Magic),下面是一个最基础的 Result 写法:
enum Result<T> {
|
当 API 请求成功,回调便会获得 Success
状态与能正确解析的数据 —— 在当前 FoodService
例子中,成功的状态包含着美食信息数组。如果请求失败,会返回 Failure
状态,并包含错误信息(如 400)。
FoodService
的 get
方法(发起 API 请求)通常会在 ViewController 中调用,ViewController 来决定请求成功失败后具体的操作逻辑:
// FoodLaLaViewController
|
但,这样处理有个问题…
关于 ViewController 中 getFood()
方法的问题是:ViewController 太过依赖这个方法了。如果没有正确的发起 API 请求或者请求结果(无论 Success
还是 Failure
)没有正确的处理,那么界面上就没有任何数据显示。
为了确保这个方法没问题,给它写测试显得尤为重要(如果实习生或者你自己以后一不小心改了什么,那界面上就啥都显示不出来了)。是的,View Controller Tests ??!
说实话,它没那么麻烦。这有一个黑魔法来配置 View Controller 测试。
OK,现在已经准备好进行 View Controller 测试了,下一步要做什么?!
为了正确地测试 ViewController 中 getFood()
方法,我们需要注入 FoodService
(依赖),而不是直接调用这个方法!
// FoodLaLaViewController
|
下面的方法便可开始测试:
// FoodLaLaViewControllerTests
|
接下来,我们需要对 FoodService
返回值类型进行更多的约束。
目前 FoodService
的结构体是这样:
struct FoodService {
|
为了方便测试,我们需要能够重写 get
方法,来控制哪个 Result(Success
或 Failure
)传给 ViewController,之后就可以测试 ViewController 是如何处理这两种结果。
因为 FoodService
是结构体类型,所以不能对其子类化。但是,你猜怎样,我们可以使用协议来达到重写目的。
我们可以将功能性代码单独提到一个协议中:
protocol Gettable {
|
注意这里标明了引用类型(associated type)。这个协议将会用在所有的 service 结构体上,现在我们只让FoodService
去遵循,但是以后还会有 CakeService
或者 DonutService
去遵循。通过使用这个通用性的协议,就可以在 App 中非常完美的统一所有 service 了。
现在,唯一需要改变的就是 FoodService
—— 让它遵循 Gettable
协议:
struct FoodService: Gettable {
|
这样写还有一个好处 —— 良好的可读性。看到 FoodService
时,你会立刻注意到 Gettable
协议。你也可以创建类似的 Creatable
,Updatable
,Delectable
,这样,service 能做的事情显而易见!
是时候重构一下了!在 ViewController 中,相比之前直接调用 FoodService
的 getFood
方法,我们现在可以将Gettable
的引用类型限制为 [Food]
。
// FoodLaLaViewController
|
现在,测试起来容易多了!
要测试 ViewController 的 getFood
方法,我们需要注入遵循 Gettable
并且引用类型为 [Food]
的 service:
// FoodLaLaViewControllerTests
|
所以,我们可以注入 Fake_FoodService
来测试 ViewController 的确发起了请求,并正确的返回了 [Food]
类型的结果(定义为 [Food]
是因为 TableView 的 data source 所要用到的类型就是 [Food]
):
// FoodLaLaViewControllerTests
|
现在你也可以仿照这个写法完成失败状态的测试(比如,根据收到的 ErrorType
显示对应的错误信息)。
使用协议来封装网络层,可以使代码统一、 可注入、 可测试、更可读。
POP 万岁!
转载自:http://swift.gg/2016/06/03/protocol-oriented-networking-in-swift/
参考链接:https://www.natashatherobot.com/protocol-oriented-networking-in-swift/
标签:
原文地址:http://www.cnblogs.com/Jenaral/p/5650597.html