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 就能毁掉整个测试.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,569评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,499评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,271评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,087评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,474评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,670评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,911评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,636评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,397评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,607评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,093评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,418评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,074评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,092评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,865评论 0 196
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,726评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,627评论 2 270

推荐阅读更多精彩内容