iOS 持续后台定位的优化方案

作者: zerojian | 来源:发表于2016-05-01 00:41 被阅读976次

    原文发表于简书:
    http://www.jianshu.com/p/5446e2580031
    作者 zerojian

    前段时间刚上线一款位置提醒软件,为了保证提醒的准确性,使用了持续后台定位,大家都知道,使用后台定位的应用,电量损耗一直是一个很棘手的问题,如果用户发现电量损耗过大,肯定会毫不犹豫的卸载你的应用。

    因为 iOS 的后台机制限制,如果需要在后台获取位置,是需要打开 Background Modes-Location updates,但必须确保定位一直处于 updateing 状态,这会导致电量损耗明显,一旦停止定位,系统会马上关闭你的 app 的后台权限。如果不是一个导航应用,不是时时刻刻需要用户的位置,那么如何才能在后台需要位置的时候获取到数据呢?也就是说,我们首先必须保证后台的持续运行,才能在这个基础上优化。

    因为如果在 Background Mode 中定位停止,那么系统会马上关闭你的后台权限,所以我们要做的就是用其它不耗电的后台方式替代后台定位,在需要用户位置的时候恢复后台定位即可。

    iOS 有一个 api BackgroundTask 它可以给程序提供最大大约3分钟的后台运行时间,它不需要在 Background Modes 申请,它一般是在程序退出后,程序如果还有事务没有处理完,那么可以申请一个短暂后台权限处理。

    我们思考一个问题,在后台定位模式中,如果在后台停止定位,是不是可以理解成普通 app 的退出程序(失去运行权限然后休眠),那么我们在后台停止定位前申请一个 BackgroundTaskTask 时间快到的时候恢复定位不就可以了吗?

    既然有这么一个方法,那么我们可以在后台每隔3分钟定位一次,而且没有任何副作用,因为 BackgroundTask 只是可以让系统给你最多3分钟后台权限,至于你要做什么,它不关心。

    要申请一个 BackgroundTask 必须有一个唯一标示符, 在申请前我们需要声明一个 Task 的 Identifier:

    var taskInvalid = UIBackgroundTaskInvalid
    

    然后我们申请一个 BackgroundTask:

    taskInvalid = UIApplication.sharedApplication().beginBackgroundTaskWithExpirationHandler({ [unowned self] in
      print("Background Task time over")
      self.endBackgroundTask()
    })
    

    需要注意的是在申请 Task 的闭包,是在 Task 系统给你的最大时间结束时(一般3分钟左右)调用的,而不是申请后调用,所以在这里面我们必须调用结束 Task

    在 Task 申请后需要处理的方法,我们可以放在 Task 申请后面,如:

      taskInvalid =   UIApplication.sharedApplication().beginBackgroundTaskWithExpiratio  nHandler({ [unowned self] in
      print("Background Task time over")
      self.endBackgroundTask()
    })
      LocationService.stopLocation()
    

    也就是说我们在申请 Task 后停止定位,由 Task 代替定位继续保持后台状态,我们在3分钟后重新开启定位,然后结束 Task ,然后重新申请一个 Task Identifier 下次使用.

    func endBackgroundTask() {
    UIApplication.sharedApplication().endBackgroundTask(taskInvalid)
    taskInvalid = UIBackgroundTaskInvalid
    print("Background task ended : \(NSDate())")
    }
    

    Task 的开启时机,我们可以使用定位次数来控制,首先我们申请一个 CLLocation 数组,然后在 didUpdateLocations 方法中每定位10次就开启 Task:

    private var backgroundLocations = [CLLocation]() {
    didSet{
      if backgroundLocations.count == 10 {
        begionBackgroundTask(self.timeInterval)
      }
    }
    }
    
     func locationManager(manager: CLLocationManager, didUpdateLocations locations: [CLLocation]) {
        guard let newLocation = locations.last else { return }
    
        if UIApplication.sharedApplication().applicationState != .Active {
      print("background location : \(newLocation.coordinate.latitude), \(newLocation.coordinate.longitude)")
      backgroundLocations.append(newLocation)
    

    至于 Task 的持续时间,我们可以使用 NSTimer 来设置:

    private func begionBackgroundTask(time: NSTimeInterval){
        initialBackgroundTask()
    
        lastBackgroundLocation?(backgroundLocations.last!)
        backgroundLocations.removeAll()
    timer = NSTimer.scheduledTimerWithTimeInterval(time, target: self, selector: #selector(againStartLocation), userInfo: nil, repeats: false)
    
    backgroundTask.registerBackgroundTask()
    
    print(" stop location :\(time) seconds")
    }
    

    这样我们就完成了定位每隔 <3分钟 工作一次,而在休息的时间内,因为没有处理任何事物,所以可以理解成休眠3分钟定位一次的电量优化方案,完整的 DEMO 可以到我的 GitHub 下载 : https://github.com/ZeroJian/ZJLocation

    转载请注明出处!

    相关文章

      网友评论

        本文标题:iOS 持续后台定位的优化方案

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