代码之家  ›  专栏  ›  技术社区  ›  JYeh

CLLocationManager startUpdatingLocation()在后台调用时不工作

  •  0
  • JYeh  · 技术社区  · 10 年前

    我有一个特定的用例,当应用程序已经在后台时,我希望能够启动位置更新。具体来说,我希望能够在我的Pebble watchapp上按下一个按钮,这会使我的同伴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__)
        }
    
    }
    
    1 回复  |  直到 10 年前
        1
  •  1
  •   scollaco    10 年前

    好吧,我不知道你到底尝试了什么,但请确保你已经打开了目标功能的位置更新(功能->背景模式),并且对于iOS 9.0,请添加以下内容:

    if (SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(@"9.0")) {
    
        self.locationManager.allowsBackgroundLocationUpdates = YES;
    }
    

    并更改行:

    locationManager.requestWhenInUseAuthorization() 
    

    收件人:

    locationManager.requestAlwaysAuthorization().
    

    祝你好运