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

原文发表于:
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 持续后台定位的优化方案)