iOS-细说单元测试(上)

  • 什么是单元测试

    单元测试,就是测试代码"单元"的功能,以确保在任何可能的条件下达到预期目的的一种测试方法.单元(Unit)是代码中一个可测试的逻辑部分. 单元测试可以帮助开发者找到错误和崩溃原因,这也是苹果拒绝上架的首要原因(crash了还上个毛架..)!

测试方法应该要能够响应所有类型的输入,包括有效输入和无效输入的情况,以确保单元能够正常运行,可以这么理解,在正常操作下要能获得我想要的结果,在异常情况(空值,缺少参数..)等一些条件下不反回对象,甚至能够对错误进行处理和回应.无论开发者对单元进行了什么更改,现有的测试方法都应该能够成功运行,而新加的测试也应该要成功运行.所以测试方法很关键!

But.很多开发者对单元测试不太感兴趣,虽然它十分整洁,可以验证许许多多的问题,但是创建,维护却需要花时间.如果要覆盖所有的功能和使用场景的话.意味着投入精力会更多一点.
小结: 测试可以增加项目的稳定性,减少错误的发生.一个良好的测试可以极高地提升用户的满意度.


  • 测试基础概念
    Xcode使用XCTest作为单元测试框架,在Xcode5之前,使用的是一个名为OCUnit的开源测试框架. XCTest就是对OCUnit的替代品,能够更好地与XCode协作.
    单元测试的概念中有四个层级:
    1. 测试套件(Test suite): 测试套件是项目中所有测试的集合,在Xcode中,测试套件作为一个独立的对象存在.
  1. 测试用类例(Test case classes): 测试功能是存放在类当中的,每个测试的例类通常是对应一个单独类来进行测试.比如: 对Login类的测试 ,应该由LoginTests类来完成,所有的单元测试类都必须要继承XCTestCase类.

  2. ** 测试用例方法():** 测试用例类包含多个方法,用来测试类的各种功能.

  3. 断言(Assertions): 断言用于检查结果是否符合预期,如果不符合,断言则会失败,并抛出失败的原因.(调试神器)

E58E1817-B9CF-473E-9F3C-F43D83A1BE51.png

创建项目的时候勾选 Include Unit Tests
成功创建之后,打开项目结构.系统帮我们生成了一个 "项目名+Tests"的文件夹.已经囊括了单元测试的 .m文件.

2EAA4ED8-4E0E-40A9-A6D8-968766AF7339.png

分析下 .m的结构

<code>
#import <XCTest/XCTest.h>
#import "Person.h"

    @interface UnitTestTests : XCTestCase

    @end

    @implementation UnitTestTests

    - (void)setUp {
      [super setUp];
     // Put setup code here. This method is called before the invocation of each test method in the class.
    }

    - (void)tearDown {
      [super tearDown];
   
      // Put setup code here. This method is called before the invocation of each test method in the class.
    }

    - (void)testExample {

     }

    - (void)testPerformanceExample {

       [self measureBlock:^{
   
       }];
    }

</code>

说说这里面的几个方法.

测试用例类 包含了 setUp 和 tearDown 方法, 这两个方法不属于测试用例方法,他们是测试用例类的初始化和析构方法.setUp里面存放的是所有对象的设置代码,而tearDown里面存放的是诸如关闭文件,取消网络请求等清理活动的代码.Xcode会依次调用 setUp,某个测试用例方法和tearDown方法.如果有多个测试方法的话,那么setUp和tearDown会在每调用一次测试方法的过程中调用一次!
testExample是示例测试方法
testPerformanceExample是性能测试的示例方法.

对于Xcode来说,测试分两种:

a. 功能测试 :功能不符合预期会报错.

b. 性能测试 :性能测试需要设置基准线,一旦发现测试结果低于基准线,或者超出最大标准差(STDDEV)限制,就会报告一个错误.
PS:这里就不对性能测试做其他说明了,有兴趣可以自行搜集资料.

注意: 自定义的测试方法必须以 test开始,这样Xcode才能找到. (左边有图标则表明方法有效)

Paste_Image.png

运行测试

当指向测试类和测试方法的时候,会出现一个按钮,我们可以运行所有的测试,也可以运行某一个独立的测试方法.测试结束后,Xcode将会返回成功或者是失败结果

Paste_Image.png

当然,我们也可以选中我们想要运行的测试类,然后选择菜单栏的 product->performAction -> Run Test Methods来运行选中的测试类.如果你只选择了一个测试类的话,那么相应的选项会变成 Run "测试方法名".

Paste_Image.png

Xcode将自动编译并运行应用程序,然后执行测试操作,测试操作完毕后,Xcode会退出应用,短暂的显示测试结果,同时测试结果会以相应的图标表示测试是否成功!
在修改好我们发现的问题之后,单击失败测试上的运行按钮,或者 选择product->performAction -> TestAgian

回到我们的例子.

Paste_Image.png

这是.m的实现文件


Paste_Image.png

简单的给person赋值了一个name,age.

目前没有对age做任何的判断.意味着不论是给age 赋值任何 int类型的 值 都能够成功!这当然不符合我们的预期.

我们在测试类里面可以开始动我们的方法了.

    //  UnitTestTests.m
    //  UnitTestTests
    //
    //  Created by uncle-R on 16/7/5.
    //  Copyright © 2016年 uncle-R. All rights reserved.
    //

    #import <XCTest/XCTest.h>
    #import "Person.h"

    @interface UnitTestTests : XCTestCase

    @property (nonatomic, strong) Person *p1;
    @end

    @implementation UnitTestTests

    - (void)setUp {
        [super setUp];
      //给p1.赋值一个 -1的age;
        self.p1 = [[Person alloc]initWithName:@"小明" andAge:-1];
        // Put setup code here. This method is called before the invocation of each test method in the class.
    }

    - (void)tearDown {
        // Put teardown code here. This method is called after the invocation of each test method in the class.
        [super tearDown];
    }

    - (void)testAge{
        //我们想要的结果是年龄不管怎样都必须大于0.
        XCTAssert(self.p1.age > 0 ,@"年龄必须大于0");

    }

    - (void)testExample {
        // This is an example of a functional test case.
        // Use XCTAssert and related functions to verify your tests produce the correct results.
        //[self ageTest];
    }

    - (void)testPerformanceExample {
        // This is an example of a performance test case.
        [self measureBlock:^{
            // Put the code you want to measure the time of here.
        }];
    }

结果如下

Paste_Image.png

可见测试结果失败.我们必须在person类的修改.

   #import "Person.h"

    @implementation Person

    -(instancetype)initWithName:(NSString*)name andAge:(int)age{
        
        if (self = [super init]) {
            //对小于0的年龄做处理
            if(age <= 0) age = 1;
            self.age = age;
            
            self.name = name;
        }
        return self;
    }

    @end

在运行一次测试.

Paste_Image.png

我们成功的通过了测试.


  • 再谈功能测试

功能测试的核心在于 "断言",上面的例子不难看出,测试结果取决于断言是否成功.
功能测试大致可以分如下几种:

1. 基础测试
2. 布尔测试
3.相等测试
4.空值测试
5.无条件失败
6.测试实例

断言:

|测试名称| 断言 | 特性 |
|: ---- :|:------:| ---- ---:|
| 基础测试 | XCTAssert | 最基础的断言,表达式为假则测试失败|
| 布尔测试 |XCTAssertTure, XCTAssertFalse | 基础测试的扩展,当表达式结果和布尔测试不匹配时则失败.
| 相等测试 | XCTAsserEqual,XCTAssertEqualObjects等 | 两个表达式不相等则测试失败 |
|空值测试 |XCTAssertNotNil |如果表达式为空则测试失败|
| 无条件失败| XCTFail | 总是测试失败(运行过这段代码就失败)|

虽然所有的测试都可以使用XCTAssert,但是Xcode还是提供了一些更有效的宏.当没有更好代替的情况下,才使用XCTAssert

格式:
** XCTAssert(表达式,消息)**

注意一下, 无条件失败通常用在控制流中,比如一个if esle结构中.if 里面都是正常的结果,else 正常情况下压根就不会进去,我们这时候可以往else丢一个 XCTFail进去,立马就能检测出异常.

当测试失败的时候

在发现测试失败的时候,先检查自己测试方法是不是存在问题,因为并不是所有逻辑一开始都想通的,因为有些时候测试失败并不仅仅是测试对象属性为nil,或者方法出问题,还可能是因为测试方法存在某些不达标的地方.
因此在测试之前,最好先想清楚测试的目的,再检查自己测试的过程是否符合要求.
tip: 当断言过多的时候要看仔细, 一个Not 就能毁掉整个测试.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容