做者:Natasha The Robot,原文连接,原文日期:2016/05/12
译者:saitjr;校对:Channe;定稿:CMBios
和我一块儿参加9 月 1 日 - 9月 2 日在纽约举办的 Swift 社区庆典?吧!使用优惠码 NATASHATHEROBOT 能够得到 $100 的折扣!git
我最近作了个 Swift 面向协议编程实践(POP?) 的演讲。视频还在处理中。另外一方面,这是演讲中 POP 视图部分的文本记录,供我和其余任何人做参考!github
假设咱们要作一款展现全球美食图片和信息的 App。这须要从 API 上拉取数据,那么,用一个对象来作网络请求也就是理所固然的了:编程
struct FoodService { func get(completionHandler: Result<[Food]> -> Void) { // 异步网络请求 // 返回请求结果 } }
一旦咱们建立了异步请求,就不能使用 Swift 內建的错误处理来同时返回成功响应和请求错误了。不过,却是给练习 Result 枚举创造了机会(更多关于 Result 枚举的信息能够参考 Error Handling in Swift: Might and Magic),下面是一个最基础的 Result 写法:swift
enum Result<T> { case Success(T) case Failure(ErrorType) }
当 API 请求成功,回调便会得到 Success
状态与能正确解析的数据 —— 在当前 FoodService
例子中,成功的状态包含着美食信息数组。若是请求失败,会返回 Failure
状态,并包含错误信息(如 400)。数组
FoodService
的 get
方法(发起 API 请求)一般会在 ViewController 中调用,ViewController 来决定请求成功失败后具体的操做逻辑:网络
// FoodLaLaViewController var dataSource = [Food]() { didSet { tableView.reloadData() } } override func viewDidLoad() { super.viewDidLoad() getFood() } private func getFood() { // 在这里调用 get() 方法 FoodService().get() { [weak self] result in switch result { case .Success(let food): self?.dataSource = food case .Failure(let error): self?.showError(error) } } }
但,这样处理有个问题...异步
关于 ViewController 中 getFood()
方法的问题是:ViewController 太过依赖这个方法了。若是没有正确的发起 API 请求或者请求结果(不管 Success
仍是 Failure
)没有正确的处理,那么界面上就没有任何数据显示。ide
为了确保这个方法没问题,给它写测试显得尤其重要(若是实习生或者你本身之后一不当心改了什么,那界面上就啥都显示不出来了)。是的,View Controller Tests ?!测试
说实话,它没那么麻烦。这有一个黑魔法来配置 View Controller 测试。
OK,如今已经准备好进行 View Controller 测试了,下一步要作什么?!
为了正确地测试 ViewController 中 getFood()
方法,咱们须要注入 FoodService
(依赖),而不是直接调用这个方法!
// FoodLaLaViewController override func viewDidLoad() { super.viewDidLoad() // 传入默认的 food service getFood(fromService: FoodService()) } // FoodService 被注入 func getFood(fromService service: FoodService) { service.get() { [weak self] result in switch result { case .Success(let food): self?.dataSource = food case .Failure(let error): self?.showError(error) } } }
下面的方法即可开始测试:
// FoodLaLaViewControllerTests func testFetchFood() { viewController.getFood(fromService: FoodService()) // ? 接下来? }
接下来,咱们须要对 FoodService
返回值类型进行更多的约束。
目前 FoodService
的结构体是这样:
struct FoodService { func get(completionHandler: Result<[Food]> -> Void) { // 发起异步请求 // 返回请求结果 } }
为了方便测试,咱们须要可以重写 get
方法,来控制哪一个 Result(Success
或 Failure
)传给 ViewController,以后就能够测试 ViewController 是如何处理这两种结果。
由于 FoodService
是结构体类型,因此不能对其子类化。可是,你猜怎样,咱们可使用协议来达到重写目的。
咱们能够将功能性代码单独提到一个协议中:
protocol Gettable { associatedtype Data func get(completionHandler: Result<Data> -> Void) }
注意这里标明了引用类型(associated type)。这个协议将会用在全部的 service 结构体上,如今咱们只让 FoodService
去遵循,可是之后还会有 CakeService
或者 DonutService
去遵循。经过使用这个通用性的协议,就能够在 App 中很是完美的统一全部 service 了。
如今,惟一须要改变的就是 FoodService
—— 让它遵循 Gettable
协议:
struct FoodService: Gettable { // [Food] 用于限制传入的引用类型 func get(completionHandler: Result<[Food]> -> Void) { // 发起异步请求 // 返回请求结果 } }
这样写还有一个好处 —— 良好的可读性。看到 FoodService
时,你会马上注意到 Gettable
协议。你也能够建立相似的 Creatable
,Updatable
,Delectable
,这样,service 能作的事情显而易见!
是时候重构一下了!在 ViewController 中,相比以前直接调用 FoodService
的 getFood
方法,咱们如今能够将 Gettable
的引用类型限制为 [Food]
。
// FoodLaLaViewController override func viewDidLoad() { super.viewDidLoad() getFood(fromService: FoodService()) } func getFood<Service: Gettable where Service.Data == [Food]>(fromService service: Service) { service.get() { [weak self] result in switch result { case .Success(let food): self?.dataSource = food case .Failure(let error): self?.showError(error) } } }
如今,测试起来容易多了!
要测试 ViewController 的 getFood
方法,咱们须要注入遵循 Gettable
而且引用类型为 [Food]
的 service:
// FoodLaLaViewControllerTests class Fake_FoodService: Gettable { var getWasCalled = false // 你也能够在这里定义一个失败结果变量,用来测试失败状态 // food 变量是一个数组(在此仅为测试目的) var result = Result.Success(food) func get(completionHandler: Result<[Food]> -> Void) { getWasCalled = true completionHandler(result) } }
因此,咱们能够注入 Fake_FoodService
来测试 ViewController 的确发起了请求,并正确的返回了 [Food]
类型的结果(定义为 [Food]
是由于 TableView 的 data source 所要用到的类型就是 [Food]
):
// FoodLaLaViewControllerTests func testFetchFood_Success() { let fakeFoodService = Fake_FoodService() viewController.getFood(fromService: fakeFoodService) XCTAssertTrue(fakeFoodService.getWasCalled) XCTAssertEqual(viewController.dataSource.count, food.count) XCTAssertEqual(viewController.dataSource, food) }
如今你也能够仿照这个写法完成失败状态的测试(好比,根据收到的 ErrorType
显示对应的错误信息)。
使用协议来封装网络层,可使代码统一、 可注入、 可测试、更可读。
POP 万岁!
本文由 SwiftGG 翻译组翻译,已经得到做者翻译受权,最新文章请访问 http://swift.gg。