首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >问答首页 >如何使用KIF框架模拟位置服务

如何使用KIF框架模拟位置服务
EN

Stack Overflow用户
提问于 2014-03-21 06:02:41
回答 2查看 766关注 0票数 6

我使用KIF框架(http://github.com/kif-framework/KIF)进行UI测试,我需要模拟位置服务。

问题是在调用KIF方法-beforeAll之前启动位置服务。所以现在嘲笑已经太晚了。

如有任何建议,将不胜感激。

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2015-02-18 14:07:08

在我的KIF目标中,我有一个BaseKIFSearchTestCase : KIFTestCase,其中覆盖了CLLocationManager`s的startUpdatingLocation。

请注意,这是我做过的唯一一个覆盖类别,因为这实际上不是一个好主意。但在测试目标上我可以接受。

代码语言:javascript
运行
AI代码解释
复制
#import <CoreLocation/CoreLocation.h>

#ifdef TARGET_IPHONE_SIMULATOR


@interface CLLocationManager (Simulator)
@end

@implementation CLLocationManager (Simulator)
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wobjc-protocol-method-implementation"

-(void)startUpdatingLocation 
{
    CLLocation *fakeLocation = [[CLLocation alloc] initWithLatitude:41.0096334 longitude:28.9651646];
    [self.delegate locationManager:self didUpdateLocations:@[fakeLocation]];
}
#pragma clang diagnostic pop

@end
#endif // TARGET_IPHONE_SIMULATOR



#import "BaseKIFSearchTestCase.h"

@interface BaseKIFSearchTestCase ()

@end

@implementation BaseKIFSearchTestCase
 //...

@end

更干净的方法是在您的应用程序目标中有一个CLLocationManager子类,在您的测试目标中有一个同名的子类,这些子类发送假位置,如上面所示。但是,如果这是可能的,则取决于您的测试目标是如何设置的,因为在Calabash使用它时,它实际上需要成为应用程序目标。

还有另一种方式:

  • 在您的项目中创建另一个配置“测试”,克隆“调试”。
  • Preprocessor Macro TESTING=1添加到该配置中。
  • 子类CLLocationManager
  • 在要使用CLLocaltionManger的地方使用该子类
  • 有条件地编译该类 #导入"GELocationManager.h“@implementation GELocationManager -(void)startUpdatingLocation { #if TESTING==1 #警告测试模式startUpdatingLocation(2*NSEC_PER_SEC),dispatch_get_main_queue(),{ CLLocation *fakeLocation = [CLLocation alloc initWithLatitude:41.0096334经度:28.9651646];[self.delegate locationManager:self didUpdateLocations:@fakeLocation];};};#;#endif }@endif}
  • 在测试目标方案中选择新的配置。

还有另一个选择:

也许是最好的:不需要修改代码。

票数 3
EN

Stack Overflow用户

发布于 2015-02-18 14:35:19

和往常一样,有几种方法可以做到这一点。关键不是试图模拟现有的位置服务,而是要拥有一个完全不同的模拟,您可以在运行时访问它。我要描述的第一个方法基本上是构建您自己的微型DI容器。第二种方法是获取您通常无法访问的单点对象。

1)重构代码,使其不直接使用LocationService。相反,将其封装在一个holder中(可以是一个简单的单例类)。然后,让你的持卡人意识到测试。它的工作方式是您有一个类似于LocationServiceHolder的东西:

代码语言:javascript
运行
AI代码解释
复制
// Do some init for your self.realService and make this holder
// a real singleton.

+ (LocationService*) locationService {
  return useMock ? self.mockService : self.realService;
}

- (void)useMock:(BOOL)useMock {
  self.useMock = useMock;
}

- (void)setMock:(LocationService*)mockService {
  self.mockService = mockService;
}

然后每当你需要你的locationService时,你就打电话给

代码语言:javascript
运行
AI代码解释
复制
[[LocationServiceHolder sharedService] locationService];  

所以当你测试的时候,你可以做这样的事情:

代码语言:javascript
运行
AI代码解释
复制
- (void)beforeAll {
  id mock = OCClassMock([LocationService class]);
  [[LocationServiceHolder sharedService] useMock:YES]];
  [[LocationServiceHolder sharedService] setMock:mock]];
}

- (void)afterAll {
  [[LocationServiceHolder sharedService] useMock:NO]];
  [[LocationServiceHolder sharedService] setMock:nil]];      
}

当然,您可以在beforeEach中这样做,并重写语义,使其比我在这里展示的基本版本要好一些。

2)如果您使用的是一个无法修改的第三方LocationService,那么它会稍微复杂一些,但仍然是可行的。这里的诀窍是使用一个类别来覆盖现有的单例方法,并公开模拟而不是普通的单例。技巧中的诀窍是,如果模拟不存在,就能够将消息发送回原始的单例。

因此,假设您有一个名为ThirdPartyService的单例。这是MockThidPartyService.h:

代码语言:javascript
运行
AI代码解释
复制
static ThirdPartyService *mockThirdPartyService;

@interface ThirdPartyService (Testing)

+ (id)sharedInstance;
+ (void)setSharedInstance:(ThirdPartyService*)instance;
+ (id)mockInstance;

@end

这是MockThidPartyService.m:

代码语言:javascript
运行
AI代码解释
复制
#import "MockThirdPartyService.h"
#import "NSObject+SupersequentImplementation.h"

// Stubbing out ThirdPartyService singleton
@implementation ThirdPartyService (Testing)

+(id)sharedInstance {
    if ([self mockInstance] != nil) {
        return [self mockInstance];
    }
    // What the hell is going on here? See http://www.cocoawithlove.com/2008/03/supersequent-implementation.html
    IMP superSequentImp = [self getImplementationOf:_cmd after:impOfCallingMethod(self, _cmd)];
    id result = ((id(*)(id, SEL))superSequentImp)(self, _cmd);
    return result;
}

+ (void)setSharedInstance:(ThirdPartyService *)instance {
    mockThirdPartyService = instance;
}

+ (id)mockInstance {
    return mockThirdPartyService;
}

@end

要使用,您可以做如下的事情:

代码语言:javascript
运行
AI代码解释
复制
#include "MockThirdPartyService.h"

...

id mock = OCClassMock([ThirdPartyService class]);
[ThirdPartyService setSharedInstance:mock];

// set up your mock and do your testing here

// Once you're done, clean up.
[ThirdPartyService setSharedInstance:nil];
// Now your singleton is no longer mocked and additional tests that
// don't depend on mock behavior can continue running.

有关超序列实现详细信息,请参阅链接。疯狂的道具为马特加拉格尔最初的想法。如果你需要的话,我也可以把文件发给你。

结论: DI是一件好事。人们抱怨必须重构和更改代码才能进行测试,但是测试可能是高质量软件开发的最重要部分,DI + ApplicationContext使事情变得非常容易。我们使用Typhoon框架,但是如果您正在进行任何级别的测试,即使您自己滚动并采用DI + ApplicationContext模式也是非常值得的。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/22560990

复制
相关文章
【Hive】从长格式表到宽格式表的转换
使用sql代码作分析的时候,几次遇到需要将长格式数据转换成宽格式数据,一般使用left join或者case when实现,代码看起来冗长,探索一下,可以使用更简单的方式实现长格式数据转换成宽格式数据。
1480
2019/05/22
2.4K0
【python】使用csv库以字典格式读写csv文件
1、使用csv.DictWriter()写入字典格式的数据 import csv with open('test.csv', 'w', newline='') as csvfile: fieldnames = ['first_name', 'last_name'] writer = csv.DictWriter(csvfile, fieldnames=fieldnames) writer.writeheader() writer.writerow({'first
西西嘛呦
2020/08/26
1.8K0
【python】使用csv库以字典格式读写csv文件
使用 Python 从字典键中删除空格
Python是广泛用于数据分析,Web开发,AI的平台,并在自动化的帮助下执行许多不同类型的任务。对我们来说,了解 python 的不同功能很重要。在本文中,我们将了解字典功能以及如何使用 python 删除键之间的空格。此功能主要用于根据需要存储和检索数据,但有时字典的键值之间可能存在空格。当用户希望访问数据时,甚至在要编辑数据的情况下,这会导致错误。
很酷的站长
2023/08/11
4960
使用 Python 从字典键中删除空格
深入Python数据分析:数据由长格式变为宽格式
melt()的逆操作在Pandas中对应为 pivot(),它也是一个设计上的顶层函数,工程位置如下:
double
2019/05/27
1.4K0
计算所汉语词法分析系统ICTCLAS 字典格式解析(字典格式说明)
     在 计算所汉语词法分析系统ICTCLAS 字典格式解析  一文中简单介绍了一下 ICTCLAS 。本来是要把字典格式一并写上去,无奈不知道怎么描述这个格式,现在终于写出了第一个Java版本的代码,也理清了思路。这个文件格式可以这样来描述:
田春峰-JCJC错别字检测
2019/02/14
5720
Python - 从字典列表中删除字典
字典是python的一个非常常用的功能,用于根据用户需要在其中存储数据。另一个典型的过程涉及编辑或操作此数据。要成为一名高效且快速的程序员,您必须弄清楚如何从字典列表中删除字典。有许多技术可以从词典列表中删除字典,本文将介绍这些技术。
很酷的站长
2023/08/11
4620
Python - 从字典列表中删除字典
【Python】json 格式转换 ① ( json 模块使用 | 列表转 json | json 转列表 | 字典转 json | json 转字典 )
然后 , 准备 python 数据 , 将数据放到 list 列表中 , 列表中的元素是 dict 字典 ;
韩曙亮
2023/10/11
7320
【Python】json 格式转换 ① ( json 模块使用 | 列表转 json | json 转列表 | 字典转 json | json 转字典 )
使用字典
字典是另一种可变容器模型,类似于我们生活中使用的字典,它可以存储任意类型对象,与列表、集合不同的是,字典的每个元素都是由一个键和一个值组成的“键值对”,键和值通过冒号分开。下面的代码演示了如何定义和使用字典。
用户8442333
2021/05/18
4480
CTF从入门到提升之宽字节注入
为什么说是从入门到放弃呢?(开个玩笑)如果说大家对CTF有了解的话,其实应该知道CTF是一个什么类型的比赛,这个比赛涉及的范围和影响有多大。如果说你真的想打好比赛,那也是真的非常不容易的,所以说这是非常困难的一件事情,初期可能学着学着就想放弃了,所以我就以这个来作为一个标题,当然本意不是让大家去放弃,就是为了让大家入个门然后再提升! 我会和我朋友一起来完成这门课程的讲解。
牛油果
2019/08/15
1.1K0
python 从字典中提取子集
prices = { 'ACME': 45.23, 'AAPL': 612.78, 'IBM': 205.55, 'HPQ': 37.20, 'FB': 10.75 } Make a dictionary of all prices over 200 p1 = {key: value for key, value in prices.items() if value > 200} Make a dictionary of tech stocks tech_names = {'AAPL', 'IB
用户5760343
2019/09/25
1.3K0
Python 字典 使用技巧
可以看出,键十从1到676,值是26个英文大写字母的组合。 我们首先要做的是找到字母组合的方法,首先想到的应该是利用chr()函数得到整型对应的字符,范围是65-90,进而可以用得到两个字符的组合,如下:
cutercorley
2020/07/23
6290
Python 字典 使用技巧
Python 之 字典使用
字典的增删改查使用 1、增加 info = { "person1":"大s", "person2":"小s", "person3":"小3", "person4":"小4" } info["person5"]="小5" info = { "广东" : { "深圳":["罗湖", "福田", "南山"], "广州":["天河", "番禺", "白云"], }, "浙江" : { "杭州":["西湖","
py3study
2020/01/14
5440
如何使用UnBlob从任意格式容器中提取文件
 关于UnBlob  UnBlob是一款针对容器安全的强大工具,该工具可以从任意格式的容器中提取文件。该工具运行速度非常快,准确率高,并且易于使用。UnBlob能够解析已知的超过30种不同格式的文档、压缩文件和文件系统,并能够从中递归提取文件内容。 UnBlob是完全开源免费的,并提供了一个命令行接口。除此之外,该工具还能够以Python库的形式来使用。这些特性使得UnBlob成为文件/数据提取、分析和逆向固件镜像的完美工具。 工具特性  1、准确率高:支持使用自定义规则识别数据区块的起始偏移量,并根
FB客服
2023/03/29
1.6K0
如何使用UnBlob从任意格式容器中提取文件
python yaml格式文件转json或字典
python代码: import yaml f = open('data.yaml', 'r') ystr = f.read() aa = yaml.load(ystr, Loader=yaml.FullLoader) print(aa) aa就是json字典对象
小贝壳
2020/03/05
4.7K0
Python读取Excel数据并以字典dict格式存储
  本文介绍基于Python语言,将一个Excel表格文件中的数据导入到Python中,并将其通过字典格式来存储的方法。
疯狂学习GIS
2023/06/26
5540
Python读取Excel数据并以字典dict格式存储
使用fold命令限制文件列宽
fold命令会从指定的文件里读取内容,将超过限定列宽的列加入增列字符后,输出到标准输出设备。若不指定任何文件名称,或是所给予的文件名为”-“,则fold指令会从标准输入设备读取数据。
用户1685462
2021/09/08
6370
Python字典使用(八)
字典的每个键值(key=>value)对用冒号分隔,每对之间用逗号分割,整个字典包括花括号在内,如下所示:
py3study
2020/01/10
7830
qtreewidget基本使用_qtreewidget列宽自适应
1、void QTreeWidget::setHeaderLabels(const QStringList & labels)
全栈程序员站长
2022/11/10
1.9K0
qtreewidget基本使用_qtreewidget列宽自适应
php宽字节注入,[投稿]宽字节注入详解
在mysql中,用于转义的函数有addslashes,mysql_real_escape_string,mysql_escape_string等,还有一种情况是magic_quote_gpc,不过高版本的PHP将去除这个特性。
全栈程序员站长
2022/11/04
3.5K0
php宽字节注入,[投稿]宽字节注入详解
MySQL数据表生成Wiki格式数据字典
<?php $dbms='mysql'; //数据库类型 $host='localhost'; //数据库主机名 $dbName='joyous'; //使用的数据库 $user='te
苦咖啡
2018/10/22
1.6K0

相似问题

Pebble - Detect Shake手势

15

Swift - Shake手势关闭子视图控制器

07

滑动手势取消IBAction的触发分段

11

将变量从手势识别器函数传递给IBAction

22

配置抽头手势识别器,以响应在点击位置的ibaction

14
添加站长 进交流群

领取专属 10元无门槛券

AI混元助手 在线答疑

扫码加入开发者社群
关注 腾讯云开发者公众号

洞察 腾讯核心技术

剖析业界实践案例

扫码关注腾讯云开发者公众号
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文