[iOS笔记]单元测试

单元测试的意义

在开发中我们需要测试我们的代码看看是否符合我们的预期,如果在每一次测试的时候我们都去找一个ViewController的viewDidLoad方法来测试,这样不仅测试代码杂乱,而且测试的时候也麻烦。这时候就需要我们使用单元测试去完成我们需要的测试。通过测试可以让我们的程序少一点Bug,及时的测试也有助于减少我们找后面找Bug的时间。

使用Xcode进行单元测试

在构建项目的时候,可以勾选下方的 Include Unit Tests,这样Xcode会帮我们自动创建单元测试

[iOS笔记]单元测试_第1张图片
新建项目.png

在项目里面直接添加Unit Test Case Class

[iOS笔记]单元测试_第2张图片
项目内添加文件.png

这样在项目中我们就能看到下面的单元测试文件

import XCTest
@testable import <#你的项目名#>

class UnitTest: XCTestCase {
    
    override func setUp() {
        super.setUp()
        // 这里放测试用例中执行前需要初始化和设置的代码
    }
    
    override func tearDown() {
        // 这里放测试用例结束后需要执行的后续操作代码
        super.tearDown()
    }
    
    // 单元测试的方法名开头必须有test,名字上最好将测试的内容都表示清楚
    func testExample() {
        // 测试的内容
        // 使用 XCTAssert 去查看测试的结果是否符合预期(类似断言的用法)
    }
    
}

可测试代码

进行单元测试的一个重要前提,就是我们要测试的代码是“可测试”(testable code)的,是指代码的测试要满足一些条件,包括:

  • 测试过程要不依赖于任何外部条件和系统;
  • 在任何环境、测试任意多次,结果应该保持不变;

依赖注入

我们管在init中可以传入其它依赖对象的方式,就叫做依赖注入(ependency Injection)。
在单元测试中,我们可以将要测试的对象抽出来,使用依赖注入来传入其他的对象来进行测试

Mock

在测试中,我们有时为了保证测试的结果保持不变,不能使用真的类型的对象,还需要创造一个假的类型使用一个假的对象来模拟。那怎么去做一个假的对象了,那就是下面使用Swift的protocol来做的Mock的思路了:

假如我们想做一个Xxx类型的MockXxx类型

  1. 定义protocol(XxxProtocol)来模拟一个原先的类型,定义出原先类型中需要测试的方法
  2. 使用extension来使原先的类型(Xxx)遵守上面的protocol(XxxProtocol)
  3. 在程序中需要测试的位置中原先的类型(Xxx)就可以改成遵守了上述protocol的类型(XxxProtocol)
  4. 创建一个Mock的类(MockXxx),遵守上述的protocol(XxxProtocol),实现里面需要测试的方法

使用上面的套路,我们就可以使用Swift的Protocol方便快捷的做出一个用于测试的假对象了

测试异步执行的回调函数

方法一

使用XCTestExpectation,简单来说,就是允许我们在一个时间范围里,给Xcode设置一个“期望”,如果期望满足了就表示测试成功,如果超时了,就表示测试失败。

// 创建 expect
let expect = expectation(description: <#描述#>)

// 在异步执行中条件满足的地方放置“满足”
expect.fulfill()

// 给“期望值”设置一个超时时间,在需要等待的地方调用
waitForExpectations(timeout: <#需要等待的秒数#>, handler: nil)

// 下面就使用 XCTAssert 去验证自己的测试

这样,只要在需要等待的秒数之内得到了返回值,测试就会成功,否则,就会失败。

通过Xcode expectation也只能部分解决我们的问题,对于测试异步执行的代码,这种方式仍有一些问题的。如果测试的是网络方法,我们无法去保证测试方法的等待秒数,而且有些网络方法有不一样的超时时间,这样这个需要等待的秒数就难以有一个统一的数字。

方法二

通过上边的mock串行化异步执行的代码:

  1. 将异步回调闭包中的属性都设成MockXxx的成员属性
  2. 在需要调用回调闭包的地方,不等待,直接调用
  3. 在单元测试中,在执行有异步回调闭包的方法之前,自己模拟给闭包中的属性设置预期值

小结:该方法就是直接跳过异步等待的时间,自己制造假的回调来使不稳定的异步执行方法变成“可测试的代码”。

你可能感兴趣的:([iOS笔记]单元测试)