gpt4 book ai didi

ios - CLLocationManager startUpdatingLocation() 在后台调用时不起作用

转载 作者:行者123 更新时间:2023-11-29 01:13:34 25 4
gpt4 key购买 nike

我有一个特定的用例,我希望能够在应用程序已在后台时启动位置更新。具体来说,我希望能够按下我的 Pebble watch 应用程序上的一个按钮,这会导致我的配套 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().

祝你好运!

关于ios - CLLocationManager startUpdatingLocation() 在后台调用时不起作用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35485215/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com