让iOS应用程序在前台运行

时间:2013-03-11 09:30:45

标签: iphone ios xcode background foreground

我想知道在特定的间隔时间内是否有办法让应用程序在前台运行。

我有一个线程正好在3分钟内运行,我需要用户不能离开应用程序。如果这样的事情不可能,我会问另一个问题。

我可以让用户在后台模式下设置应用程序,因为我使用applicationDidEnterBackground:(UIApplication *)application方法保存我需要的数据,但如果用户点击电源按钮,我该如何保存数据?

applicationWillTerminate:(UIApplication *)application会被调用吗?有没有解决方案?

2 个答案:

答案 0 :(得分:2)

您可能需要查看beginBackgroundTaskWithExpirationHandler,这可让您的应用请求额外的处理时间,以便在您的应用落后时完成任务。

那就是说,你要做的事情是不可取的。您应该确保您的应用可以在意外终止或失败的情况下从您的三分钟任务中恢复。例如,您的应用可能会因内存不足而意外崩溃或终止 - 在这种情况下,您将无法获得applicationWillTerminateapplicationDidEnterBackground的任何回调。

因此,我建议使用iOS的后台任务支持,以便在用户离开应用程序时在后台继续保存,同时确保系统终止您的应用程序而不发出警告您的任务是可以恢复/可以安全地重绕。

答案 1 :(得分:0)

您需要使用beginBackgroundTaskWithExpirationHandler来处理您的进程。使用此代码。

这里我从applicationDidEnterBackground调用此方法:( UIApplication *)应用程序将数据存储在SQlite中,即使应用程序输入后台或UIApplicationStateInactive(按下电源按钮)。如果application.backgroundTimeRemaining变为0,保存过程将按照IOS标准继续保持10分钟。我将发布一个本地通知,将我的应用程序调到前台以保持我的进程活着。使用此代码并自定义您的过程。

-(void)handleBackgroundSavingingProcess:(UIApplication *)application
    {
        NSLog(@"background task remaining time before background %f",application.backgroundTimeRemaining);

        if ((([application applicationState] == UIApplicationStateBackground) || ([application applicationState] == UIApplicationStateInactive)) && [application respondsToSelector:@selector(beginBackgroundTaskWithExpirationHandler:)])
  {


     self.bgTaskId = [application beginBackgroundTaskWithExpirationHandler:^{
     // Synchronize the cleanup call on the main thread in case
     // the task actually finishes at around the same time.
                NSLog(@"background task remaining time in expiration handler %f",application.backgroundTimeRemaining);

                dispatch_queue_t concurrentQueue;
                if([[[UIDevice currentDevice] systemVersion] doubleValue] >= 5.0)
                    concurrentQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0UL);
                else
                    concurrentQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW, 0UL);

                if (concurrentQueue == nil)
                    return;

                dispatch_async(concurrentQueue, ^{

                    if (self.bgTaskId != UIBackgroundTaskInvalid){
                        NSLog(@"background task remaining time in dispatch queue %f",application.backgroundTimeRemaining);
                        NSLog(@" Downloading status %d",self.isDownloadingInComplete);
                        if(self.isDownloadingInComplete)
                        {
                            [application presentLocalNotificationNow:localNotification];
                            NSLog(@"Local notification fired.");
                        }
                        [DataManager managedObjectContext] save:nil];
                        [application endBackgroundTask:self.bgTaskId];
                        self.bgTaskId = UIBackgroundTaskInvalid;
                        NSLog(@"Initialization invalid.");
                    }
                });
            }];
        }

    }