浅谈iOS单元测试

学习本篇文章你可以收获以下知识点:
1.单元测试简介
2.苹果自带的XCTest
3.单元测试的适用情况以及覆盖率
4.依赖注入

一.单元测试简介

单元测试是指开发者编写代码,去验证被测代码是否正确的一种手段,其实就是用代码去检测代码。合理的利用单元测试可以提高软件的质量。
我是去年开始关注单元测试这一块,并且在项目中一直在实践。可能之前一直更关注功能的实现,后期的测试都交给了QA,但是总会有一些QA也遗漏掉的点,bug上线了简直要gg。这里就对单元测试以及依赖注入做个总结,希望对大家有所帮助,提高你们项目的质量。

二.苹果自带的XCTest

苹果在Xcode7中集成了XCTest单元测试框架,我们可以在新建工程的时候直接勾选,如下图1。


图1.勾选单元测试

新建工程后,我们发现工程目录中多了一个单元测试demoTests的目录文件,我们可以在这里写我们的单元测试,如下图2。
图2.测试目录
假设我们有一个个人资料页面,里面有一项是年龄信息,我们就以这个年龄作为我们的一个测试内容。我们新建个人资料的测试用例类,记得选择Unit Test Case Class,如下图3。
图3.新建测试用例类

新建PersonalInformationTests.m测试类,.m中会有几个默认的方法,加注释简单解释。

#import
@interfacePersonalInformationTests :XCTestCase
@end
@implementationPersonalInformationTests

/** 单元测试开始前调用 */
- (void)setUp {
[supersetUp];
// 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.
[supertearDown];
}

/** 测试代码可以写到以test开头的方法中 并且test开头的方法左边会生成一个菱形图标,点击即可运行检测当前test方法内的代码 */
- (void)testExample {
// This is an example of a functional test case.
// Use XCTAssert and related functions to verify your tests produce the correct results.
}

/** 测试性能 */
- (void)testPerformanceExample {
// This is an example of a performance test case.
[selfmeasureBlock:^{
// Put the code you want to measure the time of here.
}];
}
@end

接下来就以个人资料的年龄age做测试。新建ResponsePersonalInformation.h 和.m文件,作为接口数据,我们就来模拟接口数据age的不同value情况,再进行单元测试。

#import <Foundation/Foundation.h>
@interface ResponsePersonalInformation : NSObject
@property (nonatomic, copy) NSString * age;
@end

在PersonalInformationTests中引入ResponsePersonalInformation.h,创建testPersonalInformationAge方法和checkAge方法:重点在于checkAge内部使用了断言XCTAssert,在平常的开发调试中可能使用NSLog打印信息进而分析比较多,但是如果逻辑很复杂并且需要打印的有很多岂不是很不方便么?断言可以更加方便我们的调试验证。
断言左边的参数是判断条件,右边是输出信息,如果左边条件不成立则输出右边的信息。这里只使用了一个最基本的断言XCTAssert,还有很多断言可以配合我们做测试工作,这篇博客都做了介绍。

- (void)testPersonalInformationAge
{
    ResponsePersonalInformation * response = [[ResponsePersonalInformation alloc] init];
    response.age = @"20";   //模拟合法年龄( 0<age<110认为是合法年龄)
    [self checkAge:response];
}
- (void)checkAge:(ResponsePersonalInformation *)response
{
    XCTAssert([response.age integerValue] >0 && [response.age integerValue] <= 110 , @"姓名不合法---0<age<=110认为是合法年龄");
}

点击运行- (void)testPersonalInformationAge方法左侧的菱形图标运行检查当前方法,会发现运行成功提示,如下图4。并且- (void)testPersonalInformationAge方法左边的菱形图标展示位绿颜色代表通过。


图4.检查通过

接下来我们将- (void)testPersonalInformationAge方法内容改为如下测试用例,再次点击菱形图标看看效果如下图5。

- (void)testPersonalInformationAge
{
    ResponsePersonalInformation * response = [[ResponsePersonalInformation alloc] init];
    response.age = @"120";   //模拟非法合法年龄( 0<age<110认为是合法年龄)
    [self checkAge:response];
}
图5.检查不通过

控制台打印:

Test Suite 'Selected tests' started at 2018-04-19 15:41:40.754
Test Suite '单元测试Tests.xctest' started at 2018-04-19 15:41:40.755
Test Suite 'PersonalInformationTests' started at 2018-04-19 15:41:40.755
Test Case '-[PersonalInformationTests testPersonalInformationAge]' started.
/Users/xulin/Desktop/单元测试/单元测试/PersonalInformationTests.m:34: error: -[PersonalInformationTests testPersonalInformationAge] : (([response.age integerValue] >0 && [response.age integerValue] <= 110) is true) failed - 姓名不合法---0<age<110认为是合法年龄

以上只是讲解了XCTest的基本用法,下面给大家说一下怎样利用XCTes进行性能测试,其实性能测试主要用到的就是.m中的这个方法:

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

我们将要测量执行时间的代码放到testPerformanceExample方法内部的block中:

- (void)testPerformanceExample {
    // This is an example of a performance test case.
    [self measureBlock:^{
        NSMutableArray * mutArray = [[NSMutableArray alloc] init];
        for (int i = 0; i < 9999; i++) {
            NSObject * object = [[NSObject alloc] init];
            [mutArray addObject:object];
        }
    }];
}

我在block中写了一个for循环执行9999次,然后点击方法左边的菱形图标,接着去看控制台打印信息如下:

Test Suite 'Selected tests' started at 2018-04-19 15:47:28.750
Test Suite '单元测试Tests.xctest' started at 2018-04-19 15:47:28.751
Test Suite 'PersonalInformationTests' started at 2018-04-19 15:47:28.751
Test Case '-[PersonalInformationTests testPerformanceExample]' started.
/Users/xulin/Desktop/单元测试/单元测试/PersonalInformationTests.m:50: Test Case '-[PersonalInformationTests testPerformanceExample]' measured [Time, seconds] average: 0.007, relative standard deviation: 22.585%, values: [0.006975, 0.005990, 0.005411, 0.005087, 0.006447, 0.005386, 0.006691, 0.009266, 0.009946, 0.006949], performanceMetricID:com.apple.XCTPerformanceMetric_WallClockTime, baselineName: "", baselineAverage: , maxPercentRegression: 10.000%, maxPercentRelativeStandardDeviation: 10.000%, maxRegression: 0.100, maxStandardDeviation: 0.100

我们可以从中获取到最有价值的信息: measured [Time, seconds] average: 0.002, relative standard deviation: 4.830%, values: [0.006975, 0.005990, 0.005411, 0.005087, 0.006447, 0.005386, 0.006691, 0.009266, 0.009946, 0.006949],从这里我们可以获知在一个for循环重复的代码,程序会运行10次,取一个平均运行时间值,average: 0.007这个就是平均时间0.007秒。
现在我们知道了测量一个函数的运行时间,到底这个函数效率高不高可以使用testPerformanceExample方法,但是在这之前我们怎么测试函数性能呢?我们可以使用NSTimeInterval来做,根据时间差的打印来分析,具体用法如下代码:

- (void)testPerformanceExample {
    // This is an example of a performance test case.
    [self measureBlock:^{
        NSTimeInterval startTime = CACurrentMediaTime();
        NSMutableArray * mutArray = [[NSMutableArray alloc] init];
        for (int i = 0; i < 9999; i++) {
            NSObject * object = [[NSObject alloc] init];
            [mutArray addObject:object];
        }
        NSLog(@"%f",CACurrentMediaTime() - startTime);
    }];
}

三.单元测试的适用情况以及覆盖率

在项目中很多人都不清楚到底测试用例的覆盖率是多少才合适,所以导致有的写的非常多,比如100%。不是说写的多不好,只是有些场景不需要写测试用例反倒写了 ,比如一个函数只是一个简单的变量自增操作,如果类似这样的函数都写上测试用例,会花费开发的过多时间和精力,反而得不偿失。同时也会大大增加代码量,造成逻辑混乱。因此如何拿捏好哪些需要些测试用例哪些不需要写,也是一门艺术。例如:暴漏在.h中的方法需要写测试用例, 而那些私有方法写测试用例的优先级就要低的多了。
对于测试用例覆盖度多少合适这个话题,也是仁者见仁智者见智,其实一个软件覆盖度在50%以上就可以称为一个健壮的软件了,要达到70,80这些已经是非常难了,不过我们常见的一些第三方开源框架的测试用例覆盖率还是非常高的,让人咋舌。例如,AFNNetWorking的覆盖率高达87%,SDWebImage的覆盖率高达77%。


AFN测试用例覆盖率87%

SD测试用例覆盖率77%

四.依赖注入

依赖注入是一种分离依赖,减少耦合的技术。可以帮助写出可维护,可测试的代码。那么为什么这里要提到依赖注入呢?因为在我们的单元测试中,某些时候模块间的依赖太高,会影响我们的单元测试,合理使用依赖注入帮助我们进行测试。
先举一个:

#import "Teacher.h"
#import "Student.h"
@implementation Teacher
- (void)guideStudentRead
{
    Student * gaoStu = [[Student alloc] initWithName:@"MrGao"];
    [std read];
}
@end

上面这段代码是在Teacher类中一个guideStudentRead(指导学生读书)的方法中创建一个名字叫MrGao的学生,并且这个gaoStu开始读书。这个时候其实是有一个强依赖关系的,Teacher对Student有一个依赖。这种有强依赖关系的代码其实非常不利于单元测试,首先Teacher在该方法中需要去操作一个不受自己控制的Student对象,并且如果后期修改扩展,MrGao改为了MrHu,那这个guideStudentRead方法内部也要做相应的修改。那么我们如何一个依赖对象的方法进行修改方便单元测试呢?我们可以做出如下修改:

#import "Teacher.h"
#import "Student.h"

@interface  Teacher ()
@property (nonatomic, strong) Student * stu;
@end

@implementation Teacher

- (instancetype)initWithStudent:(Student *)student
{
    if (self = [super init]) {
        self.stu = student;
    }
    return self;
}

- (void)guideStudentRead
{

}

@end

我们现在可以通过一个属性来存储依赖对象,在Teacher的init方法中传过来一个Student对象,这个Student对象的初始化不适在Teacher类中,而是在外部就已经创建好注入到Teacher类中,从而Teacher和让Student产生依赖,这个就是我们要讲的依赖注入。这样不仅减轻了依赖,也提高代码的可测试性。注入方法有很多种,我们这里使用init注入的方法称之为构造器注入。还有其他属性注入,方法注入,环境上下文注入和抽取和重写调用注入。在OC中还有一些优秀的依赖注入开源框架,比如ObjectionTyphoon 。关于依赖注入这里就简单介绍一下,如果大家还有兴趣可以继续查阅相关依赖注入的相关文档。

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

推荐阅读更多精彩内容

  • 文章来自:http://blog.csdn.net/mj813/article/details/52451355 ...
    好大一只鹏阅读 9,188评论 2 126
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,599评论 18 139
  • 单元测试不是一个小工程,需要多用些时间才能做好,不要希望通过这个文章就能掌握单元测试,这只是一个入门,需要自己动手...
    勇不言弃92阅读 7,788评论 9 60
  • 前言 如果有测试大佬发现内容不对,欢迎指正,我会及时修改。 大多数的iOS App(没有持续集成)迭代流程是这样的...
    默默_David阅读 1,650评论 0 4
  • 代码质量的重要性不言而喻,直接影响了项目质量和团队开发效率,对于如何提高代码的质量,除了依赖开发人员本身的技术素质...
    点融黑帮阅读 3,595评论 0 15