远程通知的后台任务在片刻后暂停
Posted
技术标签:
【中文标题】远程通知的后台任务在片刻后暂停【英文标题】:Background task on remote notification suspended after a short while 【发布时间】:2016-05-19 11:12:25 【问题描述】:我正在使用带有 content-available: true 标志的远程通知来在后台以静默推送通知启动应用程序,并从远程 API 处理或获取更新。当应用程序处于前台或上次运行后处于挂起状态时,代码执行良好。
在后台测试期间,当系统根据传入的静默推送通知启动应用程序时,代码仅被部分处理,应用程序在大约 150 毫秒后快速挂起。我预计应用程序将有 30 秒的时间来处理传入的通知及其有效负载。如果我需要更多时间来处理和/或获取新数据,是否需要调整应用功能或请求后台任务?
部署目标 ios 8,在 iOS 9 上测试。Xcode 7.3.1,Swift 2.2.1。
功能:开启后台模式,模式:启用远程通知
AppDelegate
func application(application: UIApplication, didReceiveRemoteNotification userInfo: [NSObject : AnyObject], fetchCompletionHandler completionHandler: (UIBackgroundFetchResult) -> Void)
if let userInfo = userInfo as? [String: AnyObject]
// Processing of the payload is done in separate Operation class (using Operations framework)
// The completion handler is called on the end of the processing/fetching in that operation
// But in case of launching the app in the background it never reaches the call of the completion handler
let parseNotificationOperation = ParseNotificationOperation(userInfo: userInfo, fetchCompletionHandler: completionHandler)
MainService.shared.enqueueApnsOperation(parseNotificationOperation)
【问题讨论】:
【参考方案1】:马丁·科尔斯,
您可以使用 expirationHandlers 来延长后台执行时间。虽然 iOS 会为您的应用分配多少时间取决于我们无法控制的各种因素,但我们注意到它主要为我们的应用在后台执行提供了 3 分钟。
以下是实现它的方法:)
在你的 AppDelegate 中声明,
UIBackgroundTaskIdentifier backgroundTask;
当你在 didReceiveRemoteNotification 中收到 APNS 时,写下这个,
if (!backgroundTask || backgroundTask == UIBackgroundTaskInvalid)
backgroundTask = [[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:^
//do all clean up job here gets called few seconds before expiration
[[UIApplication sharedApplication] endBackgroundTask:backgroundTask];
backgroundTask = UIBackgroundTaskInvalid;
];
编辑
刚刚意识到您正在使用 swift,所以这里是 swift 中的代码 :)
在AppDelegate中声明一个叫backgroundTask的变量,
var backgroundTask : UIBackgroundTaskIdentifier = UIBackgroundTaskInvalid
并在您的 didRecieveRemoteNotification 中使用它,如下所示,
func application(application: UIApplication, didReceiveRemoteNotification userInfo: [NSObject : AnyObject], fetchCompletionHandler completionHandler: (UIBackgroundFetchResult) -> Void)
if let userInfo = userInfo as? [String: AnyObject]
let parseNotificationOperation = ParseNotificationOperation(userInfo: userInfo, fetchCompletionHandler: completionHandler)
MainService.shared.enqueueApnsOperation(parseNotificationOperation)
if (backgroundTask == UIBackgroundTaskInvalid)
backgroundTask = UIApplication.sharedApplication().beginBackgroundTaskWithExpirationHandler () -> Void in
self.endTask()
最后编写一个方法来使你的过期处理程序在你完成后失效:)
func endTask()
//do all your clean up here, this is called few seconds before the task expires.
UIApplication.sharedApplication().endBackgroundTask(backgroundTask)
backgroundTask = UIBackgroundTaskInvalid;
就是这样 :) 快乐的编码伙伴 :)
【讨论】:
@SandeepBhandari 我想在收到静默推送后触发本地通知,但是一旦我将应用程序置于后台,我就没有收到任何本地通知。你能帮我解决这个问题吗? 苹果这次是否允许后台任务静默通知作弊? @digs :对不起,我还没有用 iOS 13 的变化更新我自己,所以现在还不知道:|如果我在这方面有所了解,会及时通知你以上是关于远程通知的后台任务在片刻后暂停的主要内容,如果未能解决你的问题,请参考以下文章