iOS 单元测试之常用框架 OCMock 详解

目录

前言:

一、单元测试

1.1 单元测试的必要性

1.2 单元测试的目的

- 约束条件是否通过形式参数来传送。

1.3 单元测试依赖的两个主要框架

二、OCMock 的集成与使用

2.1 OCMock 的集成方式

2.2 OCMock 的使用方法

2.3 mock 使用限制


前言:

OCMock 是一个 iOS 单元测试框架,它可以帮助开发者轻松地模拟对象和方法,从而简化单元测试的编写和维护。 

一、单元测试

1.1 单元测试的必要性

测试驱动开发并不是一个很新鲜的概念了。在日常开发中,很多时候需要测试,但是这种输出是必须在点击一系列按钮之后才能在屏幕上显示出来的东西。测试的时候,往往是用模拟器一次一次的从头开始启动 app,然后定位到自己所在模块的程序,做一系列的点击操作,然后查看结果是否符合自己预期。

这种行为无疑是对时间的巨大浪费。于是有很多资深工程师们发现,我们是可以在代码中构造一个类似的场景,然后在代码中调用我们之前想要检查的代码,并将运行结果和设想结果在程序中进行比较,如果一致,则说明我们的代码没有问题,由此就产生了单元测试。

1.2 单元测试的目的

单元测试的主要目的是发现模块内部逻辑、语法、算法和功能错误。

单元测试主要是基于白盒测试验证以下问题:

  • 验证代码与设计相符度。
  • 发现设计和需求中存在错误。
  • 发现在编码过程中引入的错误。

单元测试关注的重点有以下部分:

独立路径-对于基本执行路径和循环进行测试,可能的错误有:

  • 不同数据类型的比较。
  • “差 1 错”,即可能多循环或少循环一次。
  • 错误或不可能的终止条件。
  • 不适当的修改了循环变量。

局部数据结构-单元的局部数据结构是最常见的错误来源,应设计测试用例以检查可能的错误:

  • 不一致的数据类型。
  • 检查不正确或不一致的数据类型。

错误处理-比较完善的单元设计要能预见出错的条件,并设置适当的错误处理,以便在程序出错时,能对错误重新做安排,保证期逻辑上的正确性:

  • 出错的描述难以理解。
  • 显示的错误与实际的错误不符。
  • 对错误条件的处理不正确。

边界条件-边界上出现错误是最常见的错误现象:

  • 取最大最小值发生错误。
  • 控制流中的大于、小于这些比较值常出现错误。

单元接口-接口实际上就是输入和输出对应关系的集合,要对单元进行动态测试无非就是给这个单元一个输入,然后检查输出是否和预期一致。如果数据不能正常输入和输出,单元测试就无从谈起,因此需要对单元接口进行如下的测试:

  • 被测单元的输入、输出在个数、属性、顺序是否和详细设计中的描述一致。
  • 是否修改了只做输入用的形式参数。

- 约束条件是否通过形式参数来传送。

1.3 单元测试依赖的两个主要框架

OCUnit(即用 XCTest 进行测试)其实就是苹果自带的测试框架,主要是断言使用,由于使用简单本次文章不过多介绍。

OCMock 主要功能是模拟某个方法或者属性的返回值,你可能会疑惑为什么要这样做?使用模型生成的模型对象,再传进去不就可以了?答案是可以的,但是有特殊的情况,比如一些不容易构造或不容易获取的对象,此时你可以创建一个虚拟的对象来完成测试。实现思想是根据要 mock 的对象的 class 来创建一个对应的对象,并且设置好该对象的属性和调用预定方法后的动作(例如返回一个值,调用代码块,发送消息等等),然后将其记录到一个数组中,接下来开发者主动调用该方法,最后做一个 verify(验证),从而判断该方法是否被调用,或者调用过程中是否抛出异常等。在单元测试开发中使用更多难点的也是对 OCMock 的使用方式不明确,本次文章主要讲的就是这个 OCMock 的集成和使用方法。

二、OCMock 的集成与使用

2.1 OCMock 的集成方式

项目集成 OCMock 第三方库,这个使用 pod 工具直接安装 OCMock 框架即可。若使用 iBiu 工具安装 OCMock 库需在 podfile 文件同级创建 Podfile.custom。

使用普通的 pod 文件相同格式添加 OCmock 如下:

source 'https://github.com/CocoaPods/Specs.git'
pod 'OCMock'

2.2 OCMock 的使用方法

(一)置换方法 (存根):告诉 mock 对象,当 someMethod 被调用,返回什么值

调用方式:

d jalopy = [OCMock mockForClass[Car class]];
OCMStub([jalopy goFaster:[OCMArg any] units:@"kph"]).andReturn(@"75kph");

使用场景:

1. 验证 A 方法时,A 方法内部使用 B 方法的返回值但是 B 方法内部逻辑比较复杂,这时需要使用 stub 方法去存根 B 方法的返回值。代码实现类似下面代码实现固定 funcB 的返回值,做到在不影响源代码的条件下,获取满足测试需要的参数。

方法进行存根前

- (NSString *)getOtherTimeStrWithString:(NSString *)formatTime{NSDateFormatter *formatter = [[NSDateFormatter alloc] init];[formatter setDateStyle:NSDateFormatterMediumStyle];[formatter setTimeStyle:NSDateFormatterShortStyle];[formatter setDateFormat:@"YYYY-MM-dd HH:mm:ss"]; //(@"YYYY-MM-dd hh:mm:ss") ----------设置你想要的格式,hh与HH的区别:分别表示12小时制,24小时制//设置时区选择北京时间NSTimeZone* timeZone = [NSTimeZone timeZoneWithName:@"Asia/Beijing"];[formatter setTimeZone:timeZone];NSDate* date = [formatter dateFromString:formatTime]; //------------将字符串按formatter转成nsdate//时间转时间戳的方法:NSInteger timeSp = [[NSNumber numberWithDouble:[date timeIntervalSince1970]] integerValue] * 1000;return [NSString stringWithFormat:@"%ld",(long)timeSp];
}

使用 stub(mockObject getOtherTimeStrWithString).andReturn(@"1000") 存根后类似于以下效果

- (NSString *)getOtherTimeStrWithString:(NSString *)formatTime{return @"1000";NSDateFormatter *formatter = [[NSDateFormatter alloc] init];[formatter setDateStyle:NSDateFormatterMediumStyle];[formatter setTimeStyle:NSDateFormatterShortStyle];[formatter setDateFormat:@"YYYY-MM-dd HH:mm:ss"]; //(@"YYYY-MM-dd hh:mm:ss") ----------设置你想要的格式,hh与HH的区别:分别表示12小时制,24小时制//设置时区选择北京时间NSTimeZone* timeZone = [NSTimeZone timeZoneWithName:@"Asia/Beijing"];[formatter setTimeZone:timeZone];NSDate* date = [formatter dateFromString:formatTime]; //------------将字符串按formatter转成nsdate//时间转时间戳的方法:NSInteger timeSp = [[NSNumber numberWithDouble:[date timeIntervalSince1970]] integerValue] * 1000;return [NSString stringWithFormat:@"%ld",(long)timeSp];
}

2. 代码正常流程经过测试已经很健壮了,但是一些错误的流程并不容易发现但是是可能存在的,例如边缘值数据,单元测试中可以使用存根对数据进行模拟,测试代码在特殊数据情况下的运行情况。

注:stub() 也可以不设置返回值,验证可行,猜测可能是返回的 nil 或者 void,所以不带返回值的方法也可以进行方法存根。

(二)生成 Mock 对象,目前有三种方式。

通过对 Person 类的 talk 方法进行测试举例,其中也涉及 Men 类以及 Animaiton 类,以下是三个类的相关源码。

Person 类

@interface Person()
@property(nonatomic,strong)Men *men;
@end@implementation Person
-(void)talk:(NSString *)str
{[self.men logstr:str];[Animaiton logstr:str];}
@end

Men 类

@implementation Men
-(NSString *)logstr:(NSString *)str
{NSLog(@"%@",str);return str;
}
@end

Animaiton 类

@implementation Animaiton
+(NSString *)logstr:(NSString *)str
{NSLog(@"%@",str);return str;
}
-(NSString *)logstr:(NSString *)str
{NSLog(@"%@",str);return str;
}
@end

对 talk 方法进行单测时需要对 person 类进行 mock,以下是通过三种不同的方式生成 mock 对象,对三种方式的调用方法,使用场景都做了介绍,最后对每种方式的优缺点也做了一个表格方便区别。

Nice Mock

NiceMock 创建的 mock 对象在进行方法测试时会优先调用实例方法,若未找到实例方法,会继续调用同名的类方法。因此该方法可以用来生成 mock 对象去测试类方法也可以测试对象方法。

使用方式:

- (void)testTalkNiceMock {id mockA = OCMClassMock([Men class]);Person *person1 = [Person new];person1.men = mockA;[person1 talk:@"123"];OCMVerify([mockA logstr:[OCMArg any]]);
}

使用场景:

Nice mock 是比较友好的,当一个没有存根的方法被调用时他不会引起一个异常会验证通过。如果你不想自己对很多的方法进行存根,那么使用 nice mock。在上方的举例中 mockA 调用 testTalkNiceMock 时,Men 类中的 +(NSString *) logstr:(NSString *) str 不会执行打印操作。在调用过程中因为同时存在同名的 logstr:类方法和实例方法,会优先调用实例方法。

Strict Mock

使用方式:

测试 case 如下,mockA 是 Strict Mock 生成要调用 testTalkStrictMock 方法,则 Mock 生成要调用 testTalkStrictMock 方法则该方法要使用 stub 进行存根,否则最后的 OCMVerifyAll(mockA)就会抛出异常。

- (void)testTalkStrictMock {id mockA = OCMStrictClassMock([Person class]);OCMStub([mockA talk:@"123"]);[mockA talk:@"123"];OCMVerifyAll(mockA);
}

使用场景:

这种方式创建的 mock 对象,如果调用未 stub(stub 代表存根)的方法,会抛出一个异常。这需要保证在 mock 的生命周期中每一个独立调用的方法都是被存根的,这种方法使用比较严格,很少使用。

Partial Mock

这样创建的对象在调用方法时:如果方法被 stub,调用 stub 后的方法,如果方法没有被 stub,调用原来的对象的方法,该方法有限制只能 mock 实例对象。

使用方式:

- (void)testTalkPartialMock {id mockA = OCMPartialMock([Men new]);Person *person1 = [Person new];person1.men = mockA;[person1 talk:@"123"];OCMVerify([mockA logstr:[OCMArg any]]);
}

使用场景:

当调用一个没有被存根的方法时,会调用实际对象的该方法。当不能很好的存根一个类的方法时,该技术是非常有用的。调用 testTalkPartialMock 时 Men 类中的 +(NSString *) logstr:(NSString *) str 会执行打印操作。

三种方式的差异表格:

(三)验证方法的调用

调用方式:

OCMVerify([mock someMethod]);
OCMVerify(never(),    [mock doStuff]); //从没被调用
OCMVerify(times(n),   [mock doStuff]);   //调用了N次
OCMVerify(atLeast(n), [mock doStuff]);  //最少被调用了N次
OCMVerify(atMost(n),  [mock doStuff]);

使用场景:

在单元测试中可以验证某个方法是否执行,以及执行了几次。

延时验证调用:

OCMVerifyAllWithDelay(mock, aDelay);

使用场景:该功能用于等待异步操作会比较多,其中 aDelay 为预期最长等待时间。

(四)添加预期

调用方式:

准备数据:

NSDictionary *info = @{@"name": @"momo"};
id mock = OCMClassMock([MOOCMockDemo class]);

添加预期:

OCMExpect([mock handleLoadSuccessWithPerson:[OCMArg any]]);

可以预期不执行:

OCMReject([mock handleLoadFailWithPerson:[OCMArg any]]);

可以验证参数:

// 预期 + 参数验证
OCMExpect([mock handleLoadSuccessWithPerson:[OCMArg checkWithBlock:^BOOL(id obj) {MOPerson *person = (MOPerson *)obj;return [person.name isEqualToString:@"momo"];
}]]);

可以预期执行顺序:

// 预期下列方法顺序执行
[mock setExpectationOrderMatters:YES];
OCMExpect([mock handleLoadSuccessWithPerson:[OCMArg any]]);
OCMExpect([mock showError:NO]);

可以忽略参数 (预期方法执行时):

OCMExpect([mock showError:YES]).ignoringNonObjectArgs; // 忽视参数

执行:

[MOOCMockDemo handleLoadFinished:info];

断言:

OCMVerifyAll(mock);

可以延迟断言:

OCMVerifyAllWithDelay(mock, 1); // 支持延迟验证

最后的 OCMVerifyAll 会验证前面的期望是否有效,只要有一个没调用,就会出错。

(五)参数约束

调用方式:

OCMStub([mock someMethodWithAnArgument:[OCMArg any]])
OCMStub([mock someMethodWithPointerArgument:[OCMArg anyPointer]])
OCMStub([mock someMethodWithSelectorArgument:[OCMArg anySelector]])

使用场景:在使用 OCMVerify()方法验证某个方法是否调用是使用,单元测试会验证方法参数是否一致,如果不一致就是提示验证失败,此时如果只关注方法调用,并不关注参数即可使用 [OCMArg any] 传参。

(六)网络接口的模拟

顾名思义可以 mock 网络接口的数据返回,测试不同数据下代码的走向以及准确性。

调用方式:

id mockManager = OCMClassMock([JDStoreNetwork class]);
[orderListVc setComponentsNet:mockManager];
[OCMStub([mockManager startWithSetup:[OCMArg any] didFinish:[OCMArg any] didCancel:[OCMArg any]]) andDo:^(NSInvocation *invocation) {   void (^successBlock)(id components,NSError *error) = nil;   [invocation getArgument:&successBlock atIndex:3];  successBlock(@{@"code":@"1",@"resultCode":@"1",@"value":@{@"showOrderSearch":@"NO"}},nil);}];

以上就是在调用 setComponentsNet 方法内部调用了接口,该方法就可以在调用接口后模拟需要的返回数据,successBlock 中的就是返回的测试数据。本方式是通过获取接口调用的方法签名,获取 successBlock 成功回调传参并手动调用。同样可以模拟接口失败的情况,只需获取到签名中的对应的失败回调就可以实现了。

使用场景:书写单元测试方法时涉及网络接口的模拟,通过该方式 mock 接口返回结果。

(七)恢复类

置换类方法后,可以将类恢复到原来的状态,通过调用 stopMocking 来完成。

调用方式:

id classMock = OCMClassMock([SomeClass class]);
/* do stuff */
[classMock stopMocking];

使用场景:

正常对实例对象置换后,mock 对象释放后会自动调用 stopMocking,但是添加到类方法上的 mock 对象会跨越了多个测试,mock 的类对象在置换后不会 deallocated,需要手动来取消这个 mock 关系。

(八)观察者模拟 - 创建一个接受通知的实例

调用方式:

- (void)testPostNotification {   
Person *person1 = [[Person alloc] init];   
id observerMock = OCMObserverMock();   
//给通知中心设置观察者    
[[NSNotificationCenter defaultCenter] addMockObserver: observerMock name:@"name" object:nil];    
//设置观察期望    
[[observerMock expect] notificationWithName:@"name" object:[OCMArg any]];    //调用要验证的方法    
[person1 methodWithPostNotification];    
[[NSNotificationCenter defaultCenter] removeObserver:observerMock];    
// 调用验证   
OCMVerifyAll(observerMock);}

使用场景:

创建一个 mock 对象,可以用来观察通知。mock 必须注册以接收通知。

(九)mock 协议

调用方式:

id protocolMock = OCMProtocolMock(@protocol(SomeProtocol));
/*严格的协议*/
id classMock = OCMStrictClassMock([SomeClass class]);
id protocolMock = OCMStrictProtocolMock(@protocol(SomeProtocol));
id protocolMock = OCMProtocolMock(@protocol(SomeProtocol));
/*严格的协议*/
id classMock = OCMStrictClassMock([SomeClass class]);
id protocolMock = OCMStrictProtocolMock(@protocol(SomeProtocol));

调用场景:当需要创建一个实例,让其具有协议的所定义的功能时使用。

2.3 mock 使用限制

对于同个方法,先 stub 后 expect 是不行的:因为先 stub 的话,所有的调用都会变成 stub,这样子即使过程调用该方法,最后 OCMVerifyAll 验证也会失败;解决的办法是,在 OCMExpect 上顺便 stub,比如:OCMExpect([mock someMethod]).andReturn(@"a string"),或者将 stub 置于 expect 之后。

部分模拟不适用于某些类:如 NSString 和 NSDate,这些” toll-free bridged” 的类,否则会抛出异常。

某些方法不能 stub:如:init、class、methodSignatureForSelector、forwardInvocation 这些。

NSString 与 NSArray 的类方法不能 stub,否则无效。

NSObject 的方法调用不能验证,除非在子类中重写。

苹果核心类的私有方法调用不能被验证,如以_开头的方法。

延时验证方法调用不支持,暂时只支持期望 - 运行 - 验证模式的延时验证。

OCMock 不支持多线程。

  作为一位过来人也是希望大家少走一些弯路

在这里我给大家分享一些自动化测试前进之路的必须品,希望能对你带来帮助。

(软件测试相关资料,自动化测试相关资料,技术问题答疑等等)

相信能使你更好的进步!

点击下方小卡片

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/8199.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【unity】ShaderGraph学习笔记

【unity】ShaderGraph学习笔记 创建ShaderGraph 创建URP的shaderGraph文件 在Project面板里Create→ShaderGraph→URP→这里主要有几个选项 Lit Shader Graph:有光照三维着色器 Unlit Shader Graph:无光照三维着色器 Sprite Custom Lit Shader Gra…

CNN(卷积神经网络)的实现过程详解

概要 在图像处理领域,CNN(卷积神经网络)处于绝对统治地位,但对于CNN具体是如何用神经网络实现的,能找到的介绍要么是一大堆数学公式,要么是大段晦涩的文字说明,读起来很是辛苦,想写好一片完整的而且有深度的…

【开源项目】低代码数据可视化开发平台-Datav

Datav 基本介绍 Datav是一个Vue3搭建的低代码数据可视化开发平台,将图表或页面元素封装为基础组件,无需编写代码即可完成业务需求。 它的技术栈为:Vue3 TypeScript4 Vite2 ECharts5 Axios Pinia2 在线预览 账号: admin 密码: 123123预…

Spring Cloud+Spring Boot+Mybatis+uniapp+前后端分离实现知识付费平台免费搭建

Java版知识付费-轻松拥有知识付费平台 多种直播形式,全面满足直播场景需求 公开课、小班课、独立直播间等类型,满足讲师个性化直播场景需求;低延迟、双向视频,亲密互动,无论是互动、答疑,还是打赏、带货、…

RocketMq 事务消息原理

Rocketmq 事务消息API使用 使用TransactionMQProducer类。 实现TransactionListener 接口覆盖其方法executeLocalTransaction和checkLocalTransaction 即可。 其中executeLocalTransaction 执行本地方法和checkLocalTransaction 事务状态回查。 玩法 简历一张本地事务表&…

回归预测 | MATLAB实现TCN-BiGRU时间卷积双向门控循环单元多输入单输出回归预测

回归预测 | MATLAB实现TCN-BiGRU时间卷积双向门控循环单元多输入单输出回归预测 目录 回归预测 | MATLAB实现TCN-BiGRU时间卷积双向门控循环单元多输入单输出回归预测预测效果基本介绍模型描述程序设计参考资料 预测效果 ![6 基本介绍 1.MATLAB实现TCN-BiGRU时间卷积双向门控循…

实现基于UDP简易的英汉词典

文章目录 实现目标认识相关接口socketbzerobindrecvfromsendto 实现思路和注意事项完整代码Server.hppServer.ccClient.hppClient.cc 运行效果END 实现目标 实现一个服务端和一个客户端,客户端负责发送一个单词,服务端接收到后将翻译后的结果返回发送到…

Android 之 动画合集之帧动画

本节引言: 从本节开始我们来探究Android中的动画,毕竟在APP中添加上一些动画,会让我们的应用变得 很炫,比如最简单的关开Activity,当然自定义控件动画肯定必不可少啦~而Android中的动画 分为三大类,逐帧动画…

了解uuid

目录 一.认识 UUID 二.UUID 会耗尽吗 三.UUID 会重复吗 四.UUID 的版本 五.UUID的应用 六.java 如何生成UUID 一.认识 UUID uuid是经过特定的算法得到的. UUID 是 16 字节 128 位长的数字,通常以 36 字节的字符串表示,示例如下: 3F2…

boardmix AI:让工作效率翻倍的AI智能在线白板软件!

随着ChatGPT热度的飙升,AI逐步深入到各个领域,尤其在技术领域,引发了一场AI的新浪潮,人们谈论的焦点都与AI有关。 AI工具不仅帮助企业节约了成本,还极大提高了生产力。那些尚未融入AI的行业和产品,有着被AI…

OSI 和 TCP/IP 网络分层模型详解(基础)

OSI模型: 即开放式通信系统互联参考模型(Open System Interconnection Reference Model),是国际标准化组织(ISO)提出的一个试图使各种计算机在世界范围内互连为网络的标准框架,简称OSI。 OSI 七层模型 OS…

Windows环境部署安装Chatglm2-6B-int4

chatglm2-6B是最近比较火爆的大模型,可以在消费级显卡上部署使用,适合学习。但是一般人也不一定有那么高的硬件配置,所以部署个int4版本应该是大多数人的最好选择。我就在家里部署起了int4版本的chatglm2-6B,记录一下免得忘了。 …

【AST抽象语法树】结构分析及特性

什么是AST? AST译名抽象语法树(Abstract Syntax Tree),是一种用于表示源代码结构的数据结构。 它在编译器、解析器和静态代码分析等领域中被广泛使用。 AST结构分析 我们利用成熟的astexplorer来进行结构化的比较和分析。可以尝试登录以下网…

树 - 前缀树(Trie Tree)

树 - 前缀树(Trie Tree) 什么是前缀树前缀树的实现节点数据结构定义插入方法●非递归方式●递归方式 查询单词方法●非递归方式●递归方式 查询前缀方法●非递归方式●递归方式 前缀树的复杂度前缀树有哪些应用前缀树的压缩:基数树双数组Trie树(DoubleAr…

探讨ChatGPT的强化学习:AI学习与交互的未来

🌷🍁 博主 libin9iOak带您 Go to New World.✨🍁 🦄 个人主页——libin9iOak的博客🎐 🐳 《面试题大全》 文章图文并茂🦕生动形象🦖简单易学!欢迎大家来踩踩~&#x1f33…

ES6基础知识八:你是怎么理解ES6中Proxy的?使用场景?

一、介绍 定义: 用于定义基本操作的自定义行为 本质: 修改的是程序默认形为,就形同于在编程语言层面上做修改,属于元编程(meta programming) 元编程(Metaprogramming,又译超编程,是指某类计算…

ChatGPT和搜索引擎哪个更好用

目录 ChatGPT和搜索引擎的概念 ChatGPT和搜索引擎的作用 ChatGPT的作用 搜索引擎的作用 ChatGPT和搜索引擎哪个更好用 总结 ChatGPT和搜索引擎的概念 ChatGPT是一种基于对话的人工智能技术,而搜索引擎则是一种用于在互联网上查找和检索信息的工具。它们各自具…

kaggle新赛:Bengali.AI 语音识别大赛赛题解析

赛题名称:Bengali.AI Speech Recognition 赛题链接:https://www.kaggle.com/competitions/bengaliai-speech 赛题背景 竞赛主办方 Bengali.AI 致力于加速孟加拉语(当地称为孟加拉语)的语言技术研究。Bengali.AI 通过社区驱动的…

7p透明屏的制造过程复杂琐屑吗?

7p透明屏是一种新型的显示技术,它可以使屏幕变得透明,让用户可以透过屏幕看到背后的物体。这种技术在科幻电影中经常出现,但现在已经成为现实。 7p透明屏的工作原理是利用液晶显示技术和透明材料。液晶显示技术是一种通过控制液晶分子的排列…

【Qt】安装Qt 5.7.1 MSVC2013 64bit版本的说明

【Qt】安装Qt 5.7.1 MSVC2013 64bit版本的说明 1、背景2、安装Qt 5.7.13、运行Qt Creator 1、背景 刚开始Qt是C库,后来Qt发展就越来越强大了。后来Qt 发展成为一套跨平台C图形用户界面应用程序开发框架。 注意它不但可以开发GUI程序,而且也可用于开发非…