在Objc.io #1的Testing View Controllers中讲解的就是单元测试的相关内容。本文说下如何通过Xcode 5中集成的XCTest框架进行简单的单元测试。

什么是单元测试

首先什么是单元测试?维基百科中的解释是:

在计算机编程中,单元测试(又称为模块测试, Unit Testing)是针对程序模块(软件设计的最小单位)来进行正确性检验的测试工作。程序单元是应用的最小可测试部件。在过程化编程中,一个单元就是单个程序、函数、过程等;对于面向对象编程,最小单元就是方法,包括基类(超类)、抽象类、或者派生类(子类)中的方法。

通常来说,程序员每修改一次程序就会进行最少一次单元测试,在编写程序的过程中前后很可能要进行多次单元测试,以证实程序达到软件规格书(en:Specification)要求的工作目标,没有程序错误;虽然单元测试不是什么必须的,但也不坏,这牵涉到项目管理的政策决定。

由于我们OC程序员写的通常都是面向对象的程序,所以我们在进行单元测试时,通常都是以一个方法为单位测试,目的当然是保证其不会出错了。按照Objc.io文章的观点,如果我们代码之间的耦合度很小,那么可以将其分解成多个小部件,从而更易于测试。

原文还提到一个概念:TDD(Test-Driven Development),即测试驱动开发,该模式要求开发者在编写某个功能的代码之前先将其测试代码写好,然后编写实现代码并进行测试,从而保证实现的代码不会出现问题。因此整个项目的开发进度将由测试来驱动,这有助于开发出高质量而又正确的代码,实现敏捷开发。(Objc.io上面的文章真的非常的好)

好吧,科普完了,下面进入Xcode Unit Testing的部分。

XCTest

1.第一个单元测试

XCTest是Xcode 5中自带的测试框架,它和Xcode 4及之前的SenTestKit,OCUnit有什么前因后果,小弟没有做多少研究,所以不说了。

下面从一个Demo开始。首先用Xcode新建一个工程UnitTestDemo,工程目录结构如下:

可以看到工程下面多了一个叫UnitTestDemoTests的部分,Targets也多了一个UnitTestDemoTests,根据图标初步认为该Target跑的是一个框架。

这两个多出来的东西(相比Xcode 4没有Include Unit Tests的工程)就是用来做单元测试的,其特点是文件名或Target名都以Tests结尾。

再来看下UnitTestDemoTests.m中的代码:

#import <XCTest/XCTest.h>@interface UnitTestDemoTests : XCTestCase@end@implementation UnitTestDemoTests- (void)setUp
{[super setUp];// 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)testExample
{XCTFail(@"No implementation for \"%s\"", __PRETTY_FUNCTION__);
}@end

该类继承自XCTestCase类,其中包含 三个方法:setUp,tearDown和testExample。

setUp方法用于在测试前设置好要测试的方法,tearDown则是在测试后将设置好的要测试的方法拆卸掉。

testExample顾名思义就是一个示例。

按快捷键Command + U进行单元测试,结果如下:

可以看到没有通过测试,在Issue Navigator和控制台都输出了错误信息:本类中的testExample方法没有实现。

实际上,这个错误是我们主动抛出来的。XCTFail是一个宏,其作用就是让测试失败,后面的No implementation for \"%s\"", __PRETTY_FUNCTION__就是要报告的错误信息,由我们自定。

报错总是让人不爽,好吧,我们将其注释掉,另外写一个测试方法,尝点甜头。为了规范,建议每个测试方法都写成“ - (void)testXXX ”形式,XXX表示要测试的方法名,并且无返回类型。

//- (void)testExample
//{
//    XCTFail(@"No implementation for \"%s\"", __PRETTY_FUNCTION__);
//}- (void)testTrue {XCTAssert(1, @"Can not be zero");
}

Command + U,搞定:

注意左边的Test Navigator,绿色的标志表示测试全部通过。

2.测试的顺序

如果在同一测试类文件中多写几个方法,例如:

- (void)testTrue2 {NSLog(@"2222222222222222222222");XCTAssert(1, @"Can not be zero");
}- (void)testTrue1 {NSLog(@"1111111111111111111111");XCTAssert(1, @"Can not be zero");
}- (void)testTrue3 {NSLog(@"3333333333333333333333");XCTAssert(1, @"Can not be zero");
}- (void)testAtrue {NSLog(@"0000000000000000000000");XCTAssert(1, @"Can not be zero");
}

控制台部分输出:

Test Case '-[UnitTestDemoTests testAtrue]' started.
2014-03-19 21:19:38.182 UnitTestDemo[7401:60b] 0000000000000000000000
Test Case '-[UnitTestDemoTests testAtrue]' passed (0.001 seconds).
Test Case '-[UnitTestDemoTests testTrue1]' started.
2014-03-19 21:19:38.183 UnitTestDemo[7401:60b] 1111111111111111111111
Test Case '-[UnitTestDemoTests testTrue1]' passed (0.000 seconds).
Test Case '-[UnitTestDemoTests testTrue2]' started.
2014-03-19 21:19:38.184 UnitTestDemo[7401:60b] 2222222222222222222222
Test Case '-[UnitTestDemoTests testTrue2]' passed (0.013 seconds).
Test Case '-[UnitTestDemoTests testTrue3]' started.
2014-03-19 21:19:38.196 UnitTestDemo[7401:60b] 3333333333333333333333
Test Case '-[UnitTestDemoTests testTrue3]' passed (0.001 seconds).

可以看到无论我们怎样调换test方法的书写顺序,其测试顺序都是不变的。

目前初步的结论:测试方法执行的顺序与方法名中test后面的字符大小有关,小者优先,例如testA,testB1,testB2三个方法相继执行。

3.断言测试

下面一共18个断言(SDK中也是18个,其含义转自ios UnitTest 学习笔记,真心佩服原文的博主,部分宏小弟已经测试过):

XCTFail(format…) 生成一个失败的测试;

XCTAssertNil(a1, format...)为空判断,a1为空时通过,反之不通过;

XCTAssertNotNil(a1, format…)不为空判断,a1不为空时通过,反之不通过;

XCTAssert(expression, format...)当expression求值为TRUE时通过;

XCTAssertTrue(expression, format...)当expression求值为TRUE时通过;

XCTAssertFalse(expression, format...)当expression求值为False时通过;

XCTAssertEqualObjects(a1, a2, format...)判断相等,[a1 isEqual:a2]值为TRUE时通过,其中一个不为空时,不通过;

XCTAssertNotEqualObjects(a1, a2, format...)判断不等,[a1 isEqual:a2]值为False时通过;

XCTAssertEqual(a1, a2, format...)判断相等(当a1和a2是 C语言标量、结构体或联合体时使用,实际测试发现NSString也可以);

XCTAssertNotEqual(a1, a2, format...)判断不等(当a1和a2是 C语言标量、结构体或联合体时使用);

XCTAssertEqualWithAccuracy(a1, a2, accuracy, format...)判断相等,(double或float类型)提供一个误差范围,当在误差范围(+/-accuracy)以内相等时通过测试;

XCTAssertNotEqualWithAccuracy(a1, a2, accuracy, format...) 判断不等,(double或float类型)提供一个误差范围,当在误差范围以内不等时通过测试;

XCTAssertThrows(expression, format...)异常测试,当expression发生异常时通过;反之不通过;(很变态) XCTAssertThrowsSpecific(expression, specificException, format...) 异常测试,当expression发生specificException异常时通过;反之发生其他异常或不发生异常均不通过;

XCTAssertThrowsSpecificNamed(expression, specificException, exception_name, format...)异常测试,当expression发生具体异常、具体异常名称的异常时通过测试,反之不通过;

XCTAssertNoThrow(expression, format…)异常测试,当expression没有发生异常时通过测试;

XCTAssertNoThrowSpecific(expression, specificException, format...)异常测试,当expression没有发生具体异常、具体异常名称的异常时通过测试,反之不通过;

XCTAssertNoThrowSpecificNamed(expression, specificException, exception_name, format...)异常测试,当expression没有发生具体异常、具体异常名称的异常时通过测试,反之不通过

特别注意下XCTAssertEqualObjects和XCTAssertEqual。

XCTAssertEqualObjects(a1, a2, format...)的判断条件是[a1 isEqual:a2]是否返回一个YES。

XCTAssertEqual(a1, a2, format...)的判断条件是a1 == a2是否返回一个YES。

对于后者,如果a1和a2都是基本数据类型变量,那么只有a1 == a2才会返回YES。例如下面代码中只有第二行可以通过测试:

    // 1.比较基本数据类型变量XCTAssertEqual(1, 2, @"a1 = a2 shoud be true"); // 无法通过测试XCTAssertEqual(1, 1, @"a1 = a2 shoud be true"); // 通过测试

但是,如果a1和a2都是指针,那么只有a1和a2指向同一个对象才会返回YES。例如下面的代码中:

    // 3.比较NSArray对象NSArray *array1 = @[@1];NSArray *array2 = @[@1];NSArray *array3 = array1;XCTAssertEqual(array1, array2, @"a1 and a2 should point to the same object"); // 无法通过测试XCTAssertEqual(array1, array3, @"a1 and a2 should point to the same object"); // 通过测试

array1和array2指向不同对象,无法通过测试。

这里比较奇怪的是,NSString另当别论:

    // 2.比较NSString对象NSString *str1 = @"1";NSString *str2 = @"1";NSString *str3 = str1;XCTAssertEqual(str1, str2, @"a1 and a2 should point to the same object"); // 通过测试XCTAssertEqual(str1, str3, @"a1 and a2 should point to the same object"); // 通过测试

尽管str1和str2指向不同的对象,但是二者的指针比较却能通过测试。不知道这是不是XCTest框架本身的一个Bug,反正在这里使用NSString要小心就是了。

由于str1和str2指向同一常量,常量在内存的data段中地址是固定的,所以二者地址相同。

掌握了各个断言的含义,用起来就没什么大问题了。

4.简单的应用

说了一大堆理论和定义,下面来点实际的应用。下面有一个表格控制器:

#import "TableViewController.h"
#import "TableDataSource.h"static NSString * const kCellIdentifier = @"Cell";@interface TableViewController ()@property (strong, nonatomic) TableDataSource *dataSource;@end@implementation TableViewController
@synthesize dataSource = _dataSource;- (void)viewDidLoad
{[super viewDidLoad];TableViewCellConfigureBlock cellConfigureBlock = ^(UITableViewCell *cell, NSString *item) {cell.textLabel.text = item;};NSArray *stringsArray = @[@"1", @"2", @"3", @"1", @"2", @"3", @"1", @"2", @"3", @"1", @"2", @"3", @"1", @"2", @"3", @"1", @"2", @"3", @"1", @"2", @"3", @"1", @"2", @"3"];self.dataSource = [[TableDataSource alloc] initWithItems:stringsArrayCellIdentifier:kCellIdentifierConfigureCellBlock:cellConfigureBlock];self.tableView.dataSource = _dataSource;
}@end

该类的UITableViewDataSource委托由一个TableDataSource类实现(将UITableViewDataSource分离,见Objc.io #1Lighter View Controllers)。TableDataSource类的初始化方法如下:

- (instancetype)initWithItems:(NSArray *)anItemsCellIdentifier:(NSString *)aCellIdentifierConfigureCellBlock:(TableViewCellConfigureBlock)aConfigureCellBlock
{self = [super init];if (self) {self.items              = anItems;self.cellIdentifier     = aCellIdentifier;self.configureCellBlock = [aConfigureCellBlock copy];}return self;
}

下面写一个Tests类测试一下DataSource的初始化方法。首先新建一个test case class类:

继承自XCTestCase类:

为了规范,我们新建的测试类都应该以Tests结尾,例如CellConfigureTests。

然后写个testDataSourceInitializing方法:

- (void)testDataSourceInitializing {TableViewCellConfigureBlock cellConfigureBlock = ^(UITableViewCell *cell, NSString *item) {cell.textLabel.text = item;};TableDataSource *tableSource = [[TableDataSource alloc] initWithItems:@[@"1", @"2", @"3"]CellIdentifier:@"TestCell"ConfigureCellBlock:cellConfigureBlock];XCTAssertNotNil(tableSource, @"TableView data source should not be nil");
}

Command + U运行测试。如果TableDataSource初始化成功,那么tableSource将不会为nil,测试就能通过。

Demo下载地址:点此进入下载页

本文先说到这里,下一篇博客说下如何借助更加强大的OCMock和GHUnit进行单元测试。

参考资料:

iOS7初体验(2)——单元测试

ios UnitTest 学习笔记

XCode下的iOS单元测试

Lighter View Controllers

Testing View Controllers

单元测试

Xcode 5 单元测试(一)使用XCTest进行单元测试相关推荐

  1. xcode 5 使用 XCTest 做单元测试

    xcode 5 使用 XCTest 做单元测试 什么是单元测试,请看 百度百科 单元测试 一:在xcode5 之前,我们新建项目时,可以选择是否集成单元测试:如今在xcode5,我们新建立的项目默认就 ...

  2. maven 单元测试并行_并行运行单元测试

    maven 单元测试并行 大约是时候单元测试的开发人员能够使用批注在Parallel中运行测试. 在今天的博客文章中,我们将介绍如何使用Easytest提供的注释使传统的Junit测试并行运行. 易测 ...

  3. python单元测试框架作用_Python自动单元测试框架

    简介: 软件的测试是一件非常乏味的事情,在测试别人编写的软件时尤其如此,程序员通常都只对编写代码感兴趣,而不喜欢文档编写和软件测试这类"没有创新"的工作.既然如此,为什么不让程序员 ...

  4. @sql 单元测试_如何在SQL单元测试中使用假表?

    @sql 单元测试 In this article on SQL unit testing, we will talk about how to isolate SQL unit tests from ...

  5. 单元测试整理(一)——单元测试是什么,有什么好处

    单元测试是什么 单元测试是开发者编写的一小段代码,用于检验被测代码的一个很小的.很明确的功能是否正确,通常而言,一个单元测试是用于判断某个特定条件(或者场景)下某个特定函数的行为1. 单元测试的好处 ...

  6. 怎么写单元测试android,Android上的单元测试

    任何程序的开发都离不开单元测试来保证其健壮和稳定.Android的程序自然也不例外.从Android SDK 0.9开始,就有了比较成熟的测试框架,但是直到目前最新的1.1版本,也没有详细的文档介绍这 ...

  7. ju 单元测试_使用 JUnit4 编写单元测试

    主要内容: 从 What, Why, When, How, Deep 几个方面来介绍单元测试相关的基础知识. 需要技能: 了解 Java 编程语言,能够使用 IDEA 等. What 单元测试(uni ...

  8. java单元测试启动类配置_Springboot 单元测试简单介绍和启动所有测试类的方法

    最近一段时间都是在补之前的技术债,一直忙着写业务代码没有注重代码的质量,leader也在强求,所有要把单元测试搞起来了 我把单元测试分为两种 一个是service的单元测试,一个是controller ...

  9. python 单元测试_聊聊 Python 的单元测试框架(一):unittest

    本文首发于 HelloGitHub 公众号,并发表于 Prodesire 博客. 前言 说到 Python 的单元测试框架,想必接触过 Python 的朋友脑袋里第一个想到的就是 unittest. ...

最新文章

  1. IE6左右边框断线现象
  2. 后台提示云提醒未激活 点击激活删除方法
  3. Linux shell 编程入门 - 使用ubuntu-14.10
  4. 怎么将零件整合到一起_Fraunhofer ILT用于大型零件3D打印的“边飞行边加工”的LPBF概念...
  5. MongoDB之在mac上设置环境变量
  6. 深入理解Java反射+动态代理,java开发面试笔试题
  7. 空间数据分析与R语言实践
  8. OpenCV绘制半透明效果的代码
  9. C# 实现程序只启动一次(多次运行激活第一个实例,使其获得焦点,并在最前端显示)...
  10. ArcgisPro3.0-3.0.1中文安装包下载及安装教程
  11. tomcat8修改session的JSESSIONID名称
  12. linux更新系统内核,Linux内核升级方法详解
  13. NPU-电工电子技术第一章作业讲评
  14. x264 vbv-maxrate与vbv-bufsize对码率控制
  15. 笔记本计算机摄像头怎么打开,笔记本摄像头怎么打开,教您怎么打开笔记本的摄像头...
  16. 成都中医药大学计算机基础试题,成都中医药大学2016年春季学期期末考试.计算机基础试卷-成教(答案~)分析总结.doc...
  17. 视觉打标软件 在线视觉打标系统 1.金橙子控制板卡 2.自主研发的定位系统
  18. 2018麦考林杂志计算机科学,2018年加拿大大学麦考林杂志排名发布,快来围观你喜欢的学校排名有什么变动没?...
  19. Vivado System Generator for DSP - “Error evaluating ‘OpenFcn‘ callback of Xilinx Block“错误解决方法
  20. php工程师 英文,开发工程师的英文怎么说

热门文章

  1. bugfree安装中mysql未安装,BugFree怎么安装使用,BugFree安装使用教程
  2. 股票k线图(含具体分析过程)
  3. 音视频开发基础入门|声音的采集与量化、音频数字信号质量、音频码率
  4. Vue.js是什么?
  5. win版本caffe源码libcaffe研究
  6. 搜索dota协议服务器中,DOTA2重生BETA测试版开始
  7. 【0909】unity作业:2d箭头跟随指引物体方向,箭头始终在屏幕内。
  8. Advanced Hall Sensors Ltd
  9. 辽师836c语言真题,2020年辽宁师范大学地图学与地理信息系统考研真题试卷及试题答案,考研试题下载...
  10. linux下C语言编程操作数据库(sqlite3)