IOS 后台定位上传

作者: dacheng | 来源:发表于2016-05-18 14:27 被阅读2136次

    APP开发中经常需要使用到后台定位上传功能,在IOS中可通过CLLocationManage类来实现。其中后台定位上传可分为两类,
    第一类是Standard Location Updates,这种方法下APP在后台处于持续运行状态,同时实现定位上传。精度比较高而且可以自由调节精度,同时耗电比较大。用户可以在进入后台前关闭不必要的功能以节省电量。程序一旦被terminated,就无法上传定位信息。
    第二类是Significant Location Updates,这种方法下APP在后台处于暂停状态。当距离变化超过一定范围后,IOS系统唤醒APP并调用回调函数(函数里面可加上上传定位信息的功能)。这种方法精度稍差,不可自由调节精度,同时比较省电。即使程序被terminated,系统仍可以唤醒并触发上传定位信息。

    Standard Location Updates

    使用这种方法实现后台定位上传需要做如下配置:

    1. 在target的Capabilities选项中打开Background Modes 并勾选Location updates。如下图


      图1:Background Modes - Location updates.png

      勾选之后plist中将自动增添如下信息:


      图2: plist - Location updates.png
    2. 在plist中添加NSLocationAlwaysUsageDescription键,并在对应的value中输入提示内容,如“只有开启定位功能才能正常使用xxapp”。
      注意APP安装到iphone上后,plist中的这个信息将被显示在“隐私-定位服务”中,NSLocationAlwaysUsageDescription对应“始终”,如下图:


      图3: NSLocationAlwaysUsageDescription.png

      如果想在设置中出现上图中的“使用应用期间”选项,则应该同时在plist中添加NSLocationWhenInUseUsageDescription键并在value中输入提示内容,如下图:


      图4: plist设置.png
    3. 在程序中,必须添加如下代码(完整代码稍后提供):
      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];
      提示框将变为下图:

    允许使用应用期间定位.png

    即申请试用应用期间使用定位功能。对于这种申请用户允许的概率会高一些。如果用户选择了“Allow”,则APP同样可实现后台定位上传功能,不过在APP处于后台模式时,手机上方将出蓝色提示框,如图:

    后台定位蓝色提示框.png

    第二个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方法来完成一次定位上传。

    相关文章

      网友评论

      • 梁森的简书:想再次确认下:如果使用第二种方式定位就不用选location updates这个选项了是吗?
        第二种方式与第一种方式的区别:1.不用选location updates 2.不用代码self.standardlocationManager.allowsBackgroundLocationUpdates = YES

        是吗?
      • leftwater:第二种方式 测试被killed 没有被调用啊
      • WKCaesar:我这边勾选了location updates,也设置了locationManager.allowsBackgroundLocationUpdates = true。后台运行没有蓝色横幅。
        leftwater:加了requestAlwaysAuthorization 就不会有蓝色的条了
      • 188baaafd568:你好我这边开启一个APP后台定位不知道为啥审核被拒,作者有过这样经历!求指导。

      本文标题:IOS 后台定位上传

      本文链接:https://www.haomeiwen.com/subject/yhqwrttx.html