ios后台任务
ios里,基于设备续航和安全方面的考虑,不在前台运行的app会被系统挂起,所有线程进入沉睡态,Ios4之前,这个时机基本是即时,只要app一进入后台,程序马上就会被挂起。ios4后,允许app在后台运行任务,不过需要注册,而且有各种限制。ios7后,可以在后台运行的任务种类增加了很多,
大概罗列一下后台运行的任务类型:
- 通用类型任务
app注册后,在退后台后有一定的时间运行任务,时间到后app被挂起 - 后台播放音乐
- 位置服务
- IP电话(VoIP)
- Newsstand
ios7后新增
- 数据获取(Background Fetch)
- 推送唤醒(静默推送,Silent Remote Notifications)
- 后台传输(Background Transfer Service)
感兴趣的同学可以查看官方文档或者参考一下文章https://onevcat.com/2013/08/ios7-background-multitask/
因为最近遇到app退到后台导致上传超时失败的问题,所以专门研究了“后台传输”这个后台任务的特性和使用方式。
NSURLSession
如果app退到后台后,有数据在传输,这时候是在当前的app进程内进行的,如果app被挂起,那么线程也就会被挂起。而IOS7后,推出了NSURLSession这个类,按照官方说明,这个类的行为是由系统托管的,运行在另外的进程,不受app生命周期的影响。那就是说,即使app被挂起甚至是杀掉,系统都可以帮你完成数据传输。
一旦数据传输完成:
- 如果app在前台,那么Task和Session相关的delegate会被调用
- 如果app被挂起,那么系统先唤醒app,然后app的-application:handleEventsForBackgroundURLSession:completionHandler:方法被调用
- 如果app已经被杀死,从实际测试结果来看,什么都不会发生,只有启动app的时候,上面第二种情况的回调会被调用
限制
- 后台传输只会通过wifi来进行
- 数据传输是离散式或者说是碎片式的,因为系统可能也在运行其他的app或者任务,所以数据传输的时间是不可控的,所以代码里的超时逻辑有影响
- 如果是数据下载任务,文件是存储在系统提供的临时目录下的,而不是app的沙箱,记得要在任务完成后转移文件
使用方式
- 创建session
- (NSURLSession *)backgroundSession
{
//Use dispatch_once_t to create only one background session. If you want more than one session, do with different identifier
static NSURLSession *session = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
NSURLSessionConfiguration *configuration = [NSURLSessionConfiguration backgroundSessionConfiguration:@"com.yourcompany.appId.BackgroundSession"];
session = [NSURLSession sessionWithConfiguration:configuration delegate:self delegateQueue:nil];
});
return session;
}
2 创建任务并启动
- (void) beginDownload
{
NSURL *downloadURL = [NSURL URLWithString:DownloadURLString];
NSURLRequest *request = [NSURLRequest requestWithURL:downloadURL];
self.session = [self backgroundSession];
self.downloadTask = [self.session downloadTaskWithRequest:request];
[self.downloadTask resume];
}
3 实现delegate
AppDelegate
//AppDelegate.m
- (void)application:(UIApplication *)application handleEventsForBackgroundURLSession:(NSString *)identifier
completionHandler:(void (^)())completionHandler
{
//Check if all transfers are done, and update UI
//Then tell system background transfer over, so it can take new snapshot to show in App Switcher
completionHandler();
//You can also pop up a local notification to remind the user
//...
}
Seesion相关的delegate,这个函数在任务成功或者失败的时候被调用
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task
didCompleteWithError:(NSError *)error;
Task相关的delegate,Task有多种,比如上传任务,下载任务等,不同任务的回调会不同,比如下载任务,能获得当前下载的字节数等等
- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
didFinishDownloadingToURL:(NSURL *)location;
小结
URLSession适用于大量数据在后台传输的场景(小数据在app被挂起前应该能结束传输),不过使用中有诸多限制,对代码模块的设计冲击也比较大,需谨慎使用。