APP開發中經常需要使用到后臺定位上傳功能,在IOS中可通過CLLocationManage類來實現。其中后臺定位上傳可分為兩類,
第一類是Standard Location Updates,這種方法下APP在后臺處于持續運行狀態,同時實現定位上傳。精度比較高而且可以自由調節精度,同時耗電比較大。用戶可以在進入后臺前關閉不必要的功能以節省電量。程序一旦被terminated,就無法上傳定位信息。
第二類是Significant Location Updates,這種方法下APP在后臺處于暫停狀態。當距離變化超過一定范圍后,IOS系統喚醒APP并調用回調函數(函數里面可加上上傳定位信息的功能)。這種方法精度稍差,不可自由調節精度,同時比較省電。即使程序被terminated,系統仍可以喚醒并觸發上傳定位信息。
Standard Location Updates
使用這種方法實現后臺定位上傳需要做如下配置:
-
在target的Capabilities選項中打開Background Modes 并勾選Location updates。如下圖
圖1:Background Modes - Location updates.png
勾選之后plist中將自動增添如下信息:
圖2: plist - Location updates.png -
在plist中添加NSLocationAlwaysUsageDescription鍵,并在對應的value中輸入提示內容,如“只有開啟定位功能才能正常使用xxapp”。
注意APP安裝到iphone上后,plist中的這個信息將被顯示在“隱私-定位服務”中,NSLocationAlwaysUsageDescription對應“始終”,如下圖:
圖3: NSLocationAlwaysUsageDescription.png
如果想在設置中出現上圖中的“使用應用期間”選項,則應該同時在plist中添加NSLocationWhenInUseUsageDescription鍵并在value中輸入提示內容,如下圖:
圖4: plist設置.png -
在程序中,必須添加如下代碼(完整代碼稍后提供):
if ([[[UIDevice currentDevice] systemVersion] floatValue] >= 8) {
[self.locationManager requestAlwaysAuthorization];//在后臺也可定位
}
if ([[[UIDevice currentDevice] systemVersion] floatValue] >= 9) {
self.locationManager.allowsBackgroundLocationUpdates = YES;
}
第一個if語句是請求后臺定位功能,程序運行到這句代碼會觸發一個提示框,如下圖。
圖5: 允許后臺定位提示信息.png
如果用戶選擇“Allow”,則APP可正常后臺定位上傳。同時在隱私-設置中會出現如圖3所示結果。如果用戶選擇“Cancel”,則APP無法實現后臺定位上傳。此時只有用戶在iphone設置中手動勾選“始終”,APP才能正常后臺定位上傳。當然也可以選擇一種折中的方法,將第一個if語句中的代碼替換為
[self.standardlocationManager requestWhenInUseAuthorization];
提示框將變為下圖:
即申請試用應用期間使用定位功能。對于這種申請用戶允許的概率會高一些。如果用戶選擇了“Allow”,則APP同樣可實現后臺定位上傳功能,不過在APP處于后臺模式時,手機上方將出藍色提示框,如圖:
第二個if語句是請求后臺定位更新功能,配合上面的配置圖1使用。一旦allowsBackgroundLocationUpdates設置為YES,則上面的配置圖1是必須的,否則運行時程序會崩潰。
完成以上配之后,APP即可實現后臺定位功能,無論APP在后臺停留多久,只要不被系統由于資源不夠而殺掉,或者被用戶雙擊home 鍵移除掉,APP均能始終定位上傳。
完整的代碼如下:
//locationUpdateManager.h
#import <UIKit/UIKit.h>
#import <CoreLocation/CoreLocation.h>
#import <CoreLocation/CLLocationManager.h>
@interface locationUpdateManager:NSObject<CLLocationManagerDelegate>
+ (instancetype)sharedStandardManager;
- (void)startStandardUpdatingLocation;
- (void)stopStandardUpdatingLocation;
@end
//locationUpdateManager.m
@interface locationUpdateManager() <CLLocationManagerDelegate>
@property (strong, nonatomic) CLLocationManager *standardlocationManager;
@property (strong, nonatomic) NSDate *lastTimestamp;
@end
@implementation locationUpdateManager
# pragma mark - StandardManager
+ (instancetype)sharedStandardManager
{
static locationUpdateManager* sharedStandardInstance = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
sharedStandardInstance = [[self alloc]initStandard];
});
return sharedStandardInstance;
}
- (id)initStandard
{
if (self = [super init])
{
// 初始化工作
self.standardlocationManager = [[CLLocationManager alloc]init];
self.standardlocationManager.desiredAccuracy = kCLLocationAccuracyBest; //kCLLocationAccuracyHundredMeters better battery life
self.standardlocationManager.delegate = self;
self.standardlocationManager.pausesLocationUpdatesAutomatically = NO; // this is important
//self.standardlocationManager.distanceFilter = 10;//距離過濾
if ([[[UIDevice currentDevice] systemVersion] floatValue] >= 8) {
[self.standardlocationManager requestAlwaysAuthorization];//在后臺也可定位
}
if ([[[UIDevice currentDevice] systemVersion] floatValue] >= 9) {
self.standardlocationManager.allowsBackgroundLocationUpdates = YES;
}
}
return self;
}
- (void)startStandardUpdatingLocation
{
NSLog(@"startStandardUpdatingLocation");
[self.standardlocationManager startUpdatingLocation];
}
- (void)stopStandardUpdatingLocation
{
NSLog(@"stopStandardUpdatingLocation");
[self.standardlocationManager stopUpdatingLocation];
}
#pragma mark - 定位代理函數
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations
{
CLLocation *mostRecentLocation = locations.lastObject;
//通過socket webservice等上傳定位信息mostRecentLocation
......
}
//調用代碼
[[locationUpdateManager sharedStandardManager]startStandardUpdatingLocation];
Significant Location Updates
使用這種定位方法時,系統將默認使用基站或者WIFI定位,而不是GPS定位。這種定位方法的精度會因不同區域而異。蘋果官方給出的定位提示如下:
Apps can expect a notification as soon as the device moves 500 meters or more from its previous notification. It should not expect notifications more frequently than once every five minutes. If the device is able to retrieve data from the network, the location manager is much more likely to deliver notifications in a timely manner.
即可期待的定位精度是500米以上。不要期待比5分鐘更新一次更頻繁。
使用這種方法定位,需要Standard Location Updates中的配置2,以及配置3的第一個if語句。
APP被terminates之后,如果距離變化達到一定范圍,IOS系統會調用Appdelegate 中的application:willFinishLaunchingWithOptions: 以及application:didFinishLaunchingWithOptions:兩個函數,并傳入參數UIApplicationLaunchOptionsLocationKey,指明APP被喚醒是因為用戶位置發生了變化。喚醒之后,用戶需要重新設置一個LocationManger,并調用startMonitoringSignificantLocationChanges方法來完成一次定位上傳。