今天要实现一个需求,当用户触摸HOME键,将应用切换到后台时,启动自动备份的任务。这涉及到ios的后台任务处理,本文简单总结一下
首先,ios app有5种状态,分别是:not running, inactive, active, background, suspended,详情请看官方的guide:
如果应用处于background状态,又希望它继续做一些事的话,ios提供了几种途径:
苹果提供的PUSH机制,叫APNS。腾讯的QQ和微信就是使用这种方式。实际上,使用长连接会更好,但是苹果不支持。我理解其实应用已经suspended,但是当接收到push的数据以后,会短暂地回到background进行处理,处理完毕以后又回到suspended状态
从ios7开始,分为local push和remote push,我们应用现在还没用到,暂不深究
太玄乎了,不太了解
某些特定的任务可以在后台长时间运行,比如VOIP,location service等,只有特定类型的任务,才能用这种方式,适用性不强
类似于特定多任务,只有特殊的任务才能用
这种方式的适用性比较强,我最后也是采取这种方式来实现的。因此本文重点介绍这种
通常情况下,应用在进入background之后,很快就会转到suspended状态。但是,如果应用有需要的话(比如我们这个需求),可以向系统申请一点额外的时间来完成当前的任务
代码:
- (void)applicationDidEnterBackground:(UIApplication *)application { __block UIBackgroundTaskIdentifier bgTask;// 后台任务标识 // 结束后台任务 void (^endBackgroundTask)() = ^(){ [application endBackgroundTask:bgTask]; bgTask = UIBackgroundTaskInvalid; }; bgTask = [application beginBackgroundTaskWithExpirationHandler:^{ endBackgroundTask(); }]; dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{ double start_time = application.backgroundTimeRemaining;// 记录后台任务开始时间 BOOL networkAvailable = [YLSGlobalUtils isNetworkAvailable]; if(!networkAvailable){ NSLog(@"网络不可用,取消自动备份"); endBackgroundTask(); return; } BOOL need = [backupService checkNeedBackup]; if(!need){ NSLog(@"无需备份"); endBackgroundTask(); return; } [backupService doBackupProcessHandler:^(float done, float total){ // nothing to do with progress } CompletionHandler:^(NSError* error, NSArray* statistics){ double done_time = application.backgroundTimeRemaining; double spent_time = start_time - done_time; NSLog(@"后台备份完成,耗时: %f秒", spent_time); endBackgroundTask(); }]; }); }
- (UIBackgroundTaskIdentifier)beginBackgroundTaskWithExpirationHandler:(void(^)(void))handler
- (void)endBackgroundTask:(UIBackgroundTaskIdentifier)identifier
实际的逻辑,在下面的block里完成,这里没什么特别的,只是如果提前结束了任务,也调用一次endBackgroundTask:方法,这样就不会超时,前面的expirationHandler就不会被执行
另外,通过UIApplication的backgroundTimeRemaining属性,可以获取此后台任务还剩余的时间(当此值变成0,expirationHandler就被执行)
这段代码,是写在ApplicationDelegate的生命周期方法里:
- (void)applicationDidEnterBackground:(UIApplication *)application
- (UIBackgroundTaskIdentifier)beginBackgroundTaskWithExpirationHandler:(void(^)(void))handler
原文:http://blog.csdn.net/kyfxbl/article/details/19241201