原文发表于简书:
http://www.jianshu.com/p/5446e2580031
作者 zerojian
前段时间刚上线一款位置提醒软件,为了保证提醒的准确性,使用了持续后台定位,大家都知道,使用后台定位的应用,电量损耗一直是一个很棘手的问题,如果用户发现电量损耗过大,肯定会毫不犹豫的卸载你的应用。
因为 iOS 的后台机制限制,如果需要在后台获取位置,是需要打开 Background Modes
-Location updates
,但必须确保定位一直处于 updateing
状态,这会导致电量损耗明显,一旦停止定位,系统会马上关闭你的 app 的后台权限。如果不是一个导航应用,不是时时刻刻需要用户的位置,那么如何才能在后台需要位置的时候获取到数据呢?也就是说,我们首先必须保证后台的持续运行,才能在这个基础上优化。
因为如果在 Background Mode
中定位停止,那么系统会马上关闭你的后台权限,所以我们要做的就是用其它不耗电的后台方式替代后台定位,在需要用户位置的时候恢复后台定位即可。
iOS 有一个 api BackgroundTask
它可以给程序提供最大大约3分钟的后台运行时间,它不需要在 Background Modes
申请,它一般是在程序退出后,程序如果还有事务没有处理完,那么可以申请一个短暂后台权限处理。
我们思考一个问题,在后台定位模式中,如果在后台停止定位,是不是可以理解成普通 app 的退出程序(失去运行权限然后休眠),那么我们在后台停止定位前申请一个 BackgroundTask
在 Task
时间快到的时候恢复定位不就可以了吗?
既然有这么一个方法,那么我们可以在后台每隔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
转载请注明出处!
网友评论