ios后台网络任务NSURLSession介绍

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被挂起甚至是杀掉,系统都可以帮你完成数据传输。
一旦数据传输完成:

  1. 如果app在前台,那么Task和Session相关的delegate会被调用
  2. 如果app被挂起,那么系统先唤醒app,然后app的-application:handleEventsForBackgroundURLSession:completionHandler:方法被调用
  3. 如果app已经被杀死,从实际测试结果来看,什么都不会发生,只有启动app的时候,上面第二种情况的回调会被调用

限制

  • 后台传输只会通过wifi来进行
  • 数据传输是离散式或者说是碎片式的,因为系统可能也在运行其他的app或者任务,所以数据传输的时间是不可控的,所以代码里的超时逻辑有影响
  • 如果是数据下载任务,文件是存储在系统提供的临时目录下的,而不是app的沙箱,记得要在任务完成后转移文件

使用方式

  1. 创建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被挂起前应该能结束传输),不过使用中有诸多限制,对代码模块的设计冲击也比较大,需谨慎使用。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,671评论 8 265
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,157评论 1 32
  • 自从古老的iOS4以来,当用户点击home建的时候,你可以使你的APP们在内存中处于suspended(挂起)状态...
    木易林1阅读 3,228评论 1 4
  • 一、前言 NSURLSession是iOS7以后提出的网络请求API,这个API通过一系列的代理方法支持认证,让你...
    nuclear阅读 11,224评论 3 18
  • 今天上午老师讲了文件的保存和读取,就是把上节课练习的程序进行完善,使它可以把输入的信息保存到一个文件里,然后再次打...
    34d71ddb735c阅读 178评论 0 0