CLLocationManager startUpdatingLocation() 在后台调用时不起作用
CLLocationManager startUpdatingLocation() doesn't work when called in background
我有一个特定的用例,我希望能够在应用程序已在后台时启动位置更新。具体来说,我希望能够按下我的 Pebble 手表应用程序上的一个按钮,这会导致我的配套 iOS 应用程序开始位置更新。
仅当 iOS 应用程序处于前台或在过去几秒(如 5 秒?)内刚刚进入后台时,我才能从我的 Pebble 成功启动位置更新。应用程序在后台运行几秒钟后,我无法再开始更新。我知道更新还没有开始,因为 backgroundTimeRemaining 开始倒计时到 0(如果更新确实开始,它会保持在最大值)。我还有顶部的蓝色条,显示位置更新何时打开(“_____ 正在使用您的位置”),并且在这个奇怪的 ~5 秒阈值过去后也没有显示。
我已经查看了关于 SO 的几乎所有相关问题,并且基本上已经尝试了所建议的所有内容,例如在开始位置更新之前立即启动后台任务。我也试过在启动后台任务后延迟几秒钟开始位置更新,但这也不起作用。
奇怪的是,我还有一个 Apple Watch 应用程序,其功能与 Pebble 应用程序相同,并且 Apple Watch 能够始终无误地启动位置更新。也许 Apple Watch 可以在解除 iOS 应用程序暂停的情况下做一些特别的事情?
这是在 iOS 9.
感谢任何形式的帮助,我会尝试任何建议。
更新:我整理了一个演示该问题的快速示例应用程序。
代码贴在下面,也可以到GitHub下载工程:
https://github.com/JessicaYeh/BackgroundLocationTest
如果您尝试一下,您会发现将延迟设置为 2 秒是有效的,但当延迟时间较长(如 10 秒)时,更新不会开始。
//
// AppDelegate.swift
// BackgroundLocationTest
//
// Created by Jessica Yeh on 2/18/16.
// Copyright © 2016 Yeh. All rights reserved.
//
import UIKit
import CoreLocation
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {
var window: UIWindow?
let locationManager = CLLocationManager()
var backgroundTask: UIBackgroundTaskIdentifier = UIBackgroundTaskInvalid
var timer: NSTimer?
func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {
debugPrint(__FUNCTION__)
// Setup location manager
locationManager.delegate = self
locationManager.desiredAccuracy = kCLLocationAccuracyBest
locationManager.requestWhenInUseAuthorization()
locationManager.allowsBackgroundLocationUpdates = true
return true
}
@objc func checkBackgroundTimeRemaining() {
debugPrint(UIApplication.sharedApplication().backgroundTimeRemaining)
}
@objc func endBackgroundTask() {
if self.backgroundTask != UIBackgroundTaskInvalid {
UIApplication.sharedApplication().endBackgroundTask(self.backgroundTask)
self.backgroundTask = UIBackgroundTaskInvalid
}
}
func applicationWillResignActive(application: UIApplication) {
debugPrint(__FUNCTION__)
}
func applicationDidEnterBackground(application: UIApplication) {
debugPrint(__FUNCTION__)
// Cancel old timer and background task
timer?.invalidate()
endBackgroundTask()
// Start a new background task
backgroundTask = UIApplication.sharedApplication().beginBackgroundTaskWithExpirationHandler {
self.endBackgroundTask()
}
// Start updating location
// let delaySec = 2.0 // successfully starts location update
let delaySec = 10.0 // doesn't start location update
let delayTime = dispatch_time(DISPATCH_TIME_NOW, Int64(delaySec * Double(NSEC_PER_SEC)))
dispatch_after(delayTime, dispatch_get_main_queue(), {
debugPrint("Attempting to start updating location")
self.locationManager.startUpdatingLocation()
})
// Keep checking every 3 sec how much background time app has left
timer = NSTimer.scheduledTimerWithTimeInterval(3.0, target: self, selector: Selector("checkBackgroundTimeRemaining"), userInfo: nil, repeats: true)
}
func applicationWillEnterForeground(application: UIApplication) {
debugPrint(__FUNCTION__)
}
func applicationDidBecomeActive(application: UIApplication) {
debugPrint(__FUNCTION__)
}
func applicationWillTerminate(application: UIApplication) {
debugPrint(__FUNCTION__)
}
}
extension AppDelegate: CLLocationManagerDelegate {
func locationManager(manager: CLLocationManager, didUpdateLocations locations: [CLLocation]) {
debugPrint(__FUNCTION__)
}
}
好吧,我不完全了解您的尝试,但请确保您已打开目标功能的位置更新(功能 -> 背景模式),并且对于 iOS 9.0,添加这个:
if (SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(@"9.0")) {
self.locationManager.allowsBackgroundLocationUpdates = YES;
}
并更改行:
locationManager.requestWhenInUseAuthorization()
至:
locationManager.requestAlwaysAuthorization().
祝你好运!
我有一个特定的用例,我希望能够在应用程序已在后台时启动位置更新。具体来说,我希望能够按下我的 Pebble 手表应用程序上的一个按钮,这会导致我的配套 iOS 应用程序开始位置更新。
仅当 iOS 应用程序处于前台或在过去几秒(如 5 秒?)内刚刚进入后台时,我才能从我的 Pebble 成功启动位置更新。应用程序在后台运行几秒钟后,我无法再开始更新。我知道更新还没有开始,因为 backgroundTimeRemaining 开始倒计时到 0(如果更新确实开始,它会保持在最大值)。我还有顶部的蓝色条,显示位置更新何时打开(“_____ 正在使用您的位置”),并且在这个奇怪的 ~5 秒阈值过去后也没有显示。
我已经查看了关于 SO 的几乎所有相关问题,并且基本上已经尝试了所建议的所有内容,例如在开始位置更新之前立即启动后台任务。我也试过在启动后台任务后延迟几秒钟开始位置更新,但这也不起作用。
奇怪的是,我还有一个 Apple Watch 应用程序,其功能与 Pebble 应用程序相同,并且 Apple Watch 能够始终无误地启动位置更新。也许 Apple Watch 可以在解除 iOS 应用程序暂停的情况下做一些特别的事情?
这是在 iOS 9.
感谢任何形式的帮助,我会尝试任何建议。
更新:我整理了一个演示该问题的快速示例应用程序。 代码贴在下面,也可以到GitHub下载工程: https://github.com/JessicaYeh/BackgroundLocationTest
如果您尝试一下,您会发现将延迟设置为 2 秒是有效的,但当延迟时间较长(如 10 秒)时,更新不会开始。
//
// AppDelegate.swift
// BackgroundLocationTest
//
// Created by Jessica Yeh on 2/18/16.
// Copyright © 2016 Yeh. All rights reserved.
//
import UIKit
import CoreLocation
@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {
var window: UIWindow?
let locationManager = CLLocationManager()
var backgroundTask: UIBackgroundTaskIdentifier = UIBackgroundTaskInvalid
var timer: NSTimer?
func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {
debugPrint(__FUNCTION__)
// Setup location manager
locationManager.delegate = self
locationManager.desiredAccuracy = kCLLocationAccuracyBest
locationManager.requestWhenInUseAuthorization()
locationManager.allowsBackgroundLocationUpdates = true
return true
}
@objc func checkBackgroundTimeRemaining() {
debugPrint(UIApplication.sharedApplication().backgroundTimeRemaining)
}
@objc func endBackgroundTask() {
if self.backgroundTask != UIBackgroundTaskInvalid {
UIApplication.sharedApplication().endBackgroundTask(self.backgroundTask)
self.backgroundTask = UIBackgroundTaskInvalid
}
}
func applicationWillResignActive(application: UIApplication) {
debugPrint(__FUNCTION__)
}
func applicationDidEnterBackground(application: UIApplication) {
debugPrint(__FUNCTION__)
// Cancel old timer and background task
timer?.invalidate()
endBackgroundTask()
// Start a new background task
backgroundTask = UIApplication.sharedApplication().beginBackgroundTaskWithExpirationHandler {
self.endBackgroundTask()
}
// Start updating location
// let delaySec = 2.0 // successfully starts location update
let delaySec = 10.0 // doesn't start location update
let delayTime = dispatch_time(DISPATCH_TIME_NOW, Int64(delaySec * Double(NSEC_PER_SEC)))
dispatch_after(delayTime, dispatch_get_main_queue(), {
debugPrint("Attempting to start updating location")
self.locationManager.startUpdatingLocation()
})
// Keep checking every 3 sec how much background time app has left
timer = NSTimer.scheduledTimerWithTimeInterval(3.0, target: self, selector: Selector("checkBackgroundTimeRemaining"), userInfo: nil, repeats: true)
}
func applicationWillEnterForeground(application: UIApplication) {
debugPrint(__FUNCTION__)
}
func applicationDidBecomeActive(application: UIApplication) {
debugPrint(__FUNCTION__)
}
func applicationWillTerminate(application: UIApplication) {
debugPrint(__FUNCTION__)
}
}
extension AppDelegate: CLLocationManagerDelegate {
func locationManager(manager: CLLocationManager, didUpdateLocations locations: [CLLocation]) {
debugPrint(__FUNCTION__)
}
}
好吧,我不完全了解您的尝试,但请确保您已打开目标功能的位置更新(功能 -> 背景模式),并且对于 iOS 9.0,添加这个:
if (SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(@"9.0")) {
self.locationManager.allowsBackgroundLocationUpdates = YES;
}
并更改行:
locationManager.requestWhenInUseAuthorization()
至:
locationManager.requestAlwaysAuthorization().
祝你好运!