前言
App中越來越多的功能依賴用戶實(shí)際的位置璃诀,例如基于用戶位置提供推薦數(shù)據(jù)、基于定位判斷某些功能是否可用蒸殿,但是在開發(fā)調(diào)試中XCode卻沒有提供自定義的模擬定位的功能宾毒,所以本文主要的目的是現(xiàn)實(shí)一個(gè)可以在開發(fā)調(diào)試過程中隨時(shí)模擬定位的功能。
思路
我們在iOS的app開發(fā)中通常采用的是CLLocationManager
來獲取用戶當(dāng)前的位置盆犁,當(dāng)然也可以采用MKMapView
的showUserLocation
來獲取用戶的位置命咐,所以我們分別針對這兩種情況分析。
CLLocationManager
采用CLLocationManager
獲取定位時(shí)谐岁,是根據(jù)CLLocationManagerDelegate
中- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray<CLLocation *> *)locations
的回調(diào)來獲取到定位的醋奠。我們只需要在系統(tǒng)回調(diào)這個(gè)方法傳遞給業(yè)務(wù)代碼的中間,插入一部分代碼伊佃,來修改locations參數(shù)窜司。原本的邏輯為系統(tǒng)回調(diào)
->業(yè)務(wù)代碼
,現(xiàn)在變?yōu)?code>系統(tǒng)回調(diào)->模擬定位模塊
->業(yè)務(wù)代碼
航揉,就實(shí)現(xiàn)了無侵入式的實(shí)現(xiàn)模擬定位功能塞祈。為了實(shí)現(xiàn)這個(gè)邏輯,可以有以下幾個(gè)思路帅涂。
1议薪、 Runtime swizzle
因?yàn)?code>業(yè)務(wù)代碼是根據(jù)- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray<CLLocation *> *)locations
方法來接受回調(diào)的,所以可以采用Runtime swizzle
這個(gè)方法媳友,來實(shí)現(xiàn)模擬定位的功能斯议,但是我們中間組件是不知道業(yè)務(wù)代碼
中具體是哪個(gè)類,所以無法具體指定runtime swizzle
哪個(gè)類醇锚,所以只能遍歷所有的類哼御,判斷當(dāng)前類的方法列表中是否有locationManager:didUpdateLocations:
這個(gè)方法,如果存在則swizzle
焊唬。
- 優(yōu)點(diǎn):便于理解恋昼。
- 缺點(diǎn):需要遍歷所有的類和類的方法列表。
2求晶、中間代理對象
這種思路是Swizzle了CLLocationManager
的setDelegate:
方法焰雕,當(dāng)調(diào)用setDelegate
時(shí),將真實(shí)的delegate object
保存下來芳杏,再將我們定義的中間代理類swizzle delegate
對象設(shè)置為CLLocationManager
的delegate
,這樣當(dāng)系統(tǒng)回調(diào)CLLocationManagerDelegate
辟宗,會(huì)先回調(diào)到中間代理類swizzle delegate
中爵赵,再由swizzle delegate
將事件傳遞到真實(shí)的delegate object
。
- 優(yōu)點(diǎn):相對于第一種方法泊脐,不需要遍歷類和類的方法列表空幻,只需
swizzle
CLLocationManager
中的setDelegate:
方法即可。 - 缺點(diǎn):在中間代理類
swizzle delegate
中需要實(shí)現(xiàn)全部的CLLocationManagerDelegate
方法容客,如果后續(xù)增加代理方法秕铛,仍需要修改這個(gè)類约郁。
3、采用NSProxy實(shí)現(xiàn)中間代理對象
Objective-C中有2個(gè)基類但两,常用的就是NSObject
鬓梅,另一個(gè)就是NSProxy
,NSProxy
主要用于消息轉(zhuǎn)發(fā)處理谨湘,所以采用NSProxy
我們可以更好的處理方法二中的缺點(diǎn)绽快。
3.1
創(chuàng)建一個(gè)新的類MockLocationProxy
,集成自NSProxy
紧阔。
// MockLocationProxy.h
#import <CoreLocation/CoreLocation.h>
@interface MockLocationProxy : NSProxy
@property (nonatomic, weak, readonly, nullable) id <CLLocationManagerDelegate> target;
- (instancetype)initWithTarget:(id <CLLocationManagerDelegate>)target;
@end
// MockLocationProxy.m
#import "MockLocationProxy.h"
@implementation MockLocationProxy
- (instancetype)initWithTarget:(id<CLLocationManagerDelegate>)target {
_target = target;
return self;
}
@end
接著就來處理消息轉(zhuǎn)發(fā)的邏輯坊罢,首先我們要知道我們想要的是什么效果,系統(tǒng)回調(diào)給MockLocationProxy
擅耽,MockLocationProxy
只處理locationManager:didUpdateLocations:
活孩,其他的消息都仍然交給原target。
所以我們在MockLocationProxy.m
中添加以下方法:
// MockLocationProxy.m
@implementation MockLocationProxy
- (instancetype)initWithTarget:(id<CLLocationManagerDelegate>)target {
_target = target;
return self;
}
- (BOOL)respondsToSelector:(SEL)aSelector {
if (aSelector == @selector(locationManager:didUpdateLocations:)) {
return YES;
}
return [self.target respondsToSelector:aSelector];
}
- (void)forwardInvocation:(NSInvocation *)invocation {
SEL sel = invocation.selector;
if ([self.target respondsToSelector:sel]) {
[invocation invokeWithTarget:self.target];
}
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)sel {
return [self.target methodSignatureForSelector:sel];
}
#pragma mark - CLLocationManagerDelegate
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray<CLLocation *> *)locations {
if ([self.target respondsToSelector:_cmd]) {
// 模擬定位代碼
CLLocation *mockLocation = [[CLLocation alloc] initWithLatitude:39.908722 longitude:116.397499];
locations = @[mockLocation];
[self.target locationManager:manager didUpdateLocations:locations];
}
}
@end
當(dāng)消息發(fā)送給MockLocationProxy
時(shí)乖仇,判斷當(dāng)前方法是否是locationManager:didUpdateLocations:
憾儒,如果是,則MockLocationProxy
響應(yīng)事件这敬,否則直接傳遞給原本的target航夺。到此已經(jīng)可以隨時(shí)處理模擬定位。你只需要在模擬定位的代碼做一些處理崔涂,就可以隨時(shí)修改定位阳掐。
One more.
上述方法雖然可以模擬定位,但是每次修改模擬值都需重新build冷蚂,那么有沒有辦法在運(yùn)行時(shí)隨時(shí)修改這個(gè)值呢缭保?
LLDebugTool
當(dāng)然可以,你只需要在你的項(xiàng)目中集成LLDebugTool
蝙茶,調(diào)用其中的Location模塊艺骂,LLDebugTool
提供了一個(gè)UI來隨時(shí)修改這個(gè)模擬值,讓你在調(diào)試時(shí)隆夯,隨時(shí)模擬定位钳恕,LLDebugTool
仍提供了很多其他的功能,如果你只需要模擬定位的功能蹄衷,則只需要集成LLDebugTool/Location
這個(gè)subspec就可以了忧额。
后記
前言說過,定位除了CLLocationManager之外愧口,MKMapView
的showUserLocation
也可以獲取定位信息睦番,那么如何解決這個(gè)問題呢? 你可以在LLDebugTool/Location
中查看答案。