iOS-细说单元测试(上)

作者: uncleRX | 来源:发表于2016-07-05 17:38 被阅读891次
  • 什么是单元测试

    单元测试,就是测试代码"单元"的功能,以确保在任何可能的条件下达到预期目的的一种测试方法.单元(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 就能毁掉整个测试.

相关文章

  • iOS-细说单元测试(上)

    什么是单元测试单元测试,就是测试代码"单元"的功能,以确保在任何可能的条件下达到预期目的的一种测试方法.单元(Un...

  • 单元测试UnitTest

    来源iOS-使用Xcode自带单元测试UnitTest - 简书 - (void)setUp {[super se...

  • iOS-自动打包及分发(三)

    iOS-自动打包及分发(一)iOS-自动打包及分发(二)iOS-自动打包及分发(三) 废话不多说了,上正文: 一、...

  • iOS-单元测试

    单元测试的好处是可以在其中随意写测试代码而不会影响到主程序的功能;也许和个人所在公司和项目的原因,在实际开发中单元...

  • iOS-单元测试

    推荐链接https://www.jianshu.com/p/84ffc4f11042

  • iOS-单元测试

    单测在比较大的项目中会使用到,下面的几个东西是单测常用的工具,供大家参考。LCOV - 单测覆盖率报告生成工具;O...

  • iOS- 打包时 UUID出错的解决方案

    在这之前我们先回顾一下《iOS- 最全的真机测试教程》《iOS-最全的App上架教程》 Your build se...

  • iOS-单元测试汇总

    前言: 对于单元测试来说,我想大部分同行,在项目中,很少会用到,也有一大部分,知道单元测试这个东西,但是确切的说没...

  • iOS-单元测试汇总

    原文地址: https://www.jianshu.com/p/4001e06b150e 前言: 对于单元测试来说...

  • iOS-私有API与runtime

    iOS-私有API与runtime iOS-私有API与runtime

网友评论

  • 439dc6d129ac:现在遇到一个问题,测试一个类方法,但类方法中又调用了两个类方法和一个实例方法.测试通不过,请问大神有什么方法解决这种问题?需要测试的这个类方法是一个工具类.
  • 随风风流:单元测试能否监听按钮点击事件是否符合预期呢?如果对大项目进行单元测试是不是要自己new多个tests类呢?
    随风风流:@uncleRX 类的私有方法是不能调用的好像。
    随风风流:@uncleRX 能否在项目运行过程中进行单元测试呢?比如创建了网络请求的公共类,想要判断在访问哪个链接的时候返回的是“404”。
    uncleRX:@随风风流 可以监听按钮点击之后是否合乎预期,你可以在测试方法里面手动调用按钮的点击方法(你自己控制条件)。每一个模块都需要new一个Test 、所以项目够大会有很多
  • lucifrom_long:写得不错啊,膜拜
  • 鬼崇祟:我想请问一下,项目中所有类都要用这一个生成的test吗 ,还是说其他要自己根据需要建,怎么建?
    鬼崇祟:@uncleRX 3q
    uncleRX:@鬼崇祟 当然自己可以new很多个测试类。记得继承测试基类
  • Pocket:可以

本文标题:iOS-细说单元测试(上)

本文链接:https://www.haomeiwen.com/subject/xmvhjttx.html