使用NSURLSession

作者: CoderAO | 来源:发表于2015-07-18 01:28 被阅读47944次

    写此文时突发灵感作诗一首, 而后置顶, 欢迎品鉴.

    有的程序员老了,还没听过NSURLSession
    有的程序员还嫩,没用过NSURLConnection
    有的程序员很单纯,他只知道AFN.

    NSURLConnection在iOS9被宣布弃用,NSURLSession从13年发展到现在,终于迎来了它独步江湖的时代.NSURLSession是苹果在iOS7后为HTTP数据传输提供的一系列接口,比NSURLConnection强大,坑少,好用.今天从使用的角度介绍下.

    除了NSURLSession,文中还会频繁地出现NSURLSessionConfigurationNSURLSessionTask两个类.先认识一下,混个脸熟吧.

    使用NSURLSession,拢共分两步:

    • 第一步 通过NSURLSession的实例创建task
    • 第二部 执行task

    既然两步里面都出现了task,就先说说它吧.

    NSURLSessionTask可以简单理解为任务:如数据请求任务,下载任务,上传任务and so on.我们使用的是他的子类们:

    • NSURLSessionTask(抽象类)
      • NSURLSessionDataTask
        • NSURLSessionUploadTask
      • NSURLSessionDownloadTask

    从这几个子类的名字就可以大概猜出他们的作用了.接下来我们就从不同类型的任务出发,来使用session.

    NSURLSessionDataTask

    字面上看是和数据相关的任务,但其实dataTask完全可以胜任downloadTask和uploadTask的工作.这可能也是我们使用最多的task种类.

    简单GET请求

    如果请求的数据比较简单,也不需要对返回的数据做一些复杂的操作.那么我们可以使用带block

    // 快捷方式获得session对象
    NSURLSession *session = [NSURLSession sharedSession];
    NSURL *url = [NSURL URLWithString:@"http://www.daka.com/login?username=daka&pwd=123"];
    // 通过URL初始化task,在block内部可以直接对返回的数据进行处理
    NSURLSessionTask *task = [session dataTaskWithURL:url
                                   completionHandler:^(NSData *data, NSURLResponse *response, NSError error) {
        NSLog(@"%@", [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil]);
    }];
    
    // 启动任务
    [task resume];
    

    Tips:

    • 所有类型的task都要调用resume方法才会开始进行请求.

    简单POST请求

    POST和GET的区别就在于request,所以使用session的POST请求和GET过程是一样的,区别就在于对request的处理.

    NSURL *url = [NSURL URLWithString:@"http://www.daka.com/login"];
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
    request.HTTPMethod = @"POST";
    request.HTTPBody = [@"username=daka&pwd=123" dataUsingEncoding:NSUTF8StringEncoding];
    
    NSURLSession *session = [NSURLSession sharedSession];
    // 由于要先对request先行处理,我们通过request初始化task
    NSURLSessionTask *task = [session dataTaskWithRequest:request
                                       completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) { NSLog(@"%@", [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil]); }];
    [task resume];
    

    NSURLSessionDataDelegate代理方法

    NSURLSession提供了block方式处理返回数据的简便方式,但如果想要在接收数据过程中做进一步的处理,仍然可以调用相关的协议方法.NSURLSession的代理方法和NSURLConnection有些类似,都是分为接收响应、接收数据、请求完成几个阶段.

    // 使用代理方法需要设置代理,但是session的delegate属性是只读的,要想设置代理只能通过这种方式创建session
    NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]
                                                          delegate:self
                                                     delegateQueue:[[NSOperationQueue alloc] init]];
    
    // 创建任务(因为要使用代理方法,就不需要block方式的初始化了)
    NSURLSessionDataTask *task = [session dataTaskWithRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"http://www.daka.com/login?userName=daka&pwd=123"]]];
    
    // 启动任务
    [task resume];
    
    //对应的代理方法如下:
    
    // 1.接收到服务器的响应
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler {
        // 允许处理服务器的响应,才会继续接收服务器返回的数据
        completionHandler(NSURLSessionResponseAllow);
    }
    
    // 2.接收到服务器的数据(可能调用多次)
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data {
        // 处理每次接收的数据
    }
    
    // 3.请求成功或者失败(如果失败,error有值)
    - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
        // 请求完成,成功或者失败的处理
    }
    

    Tips:

    关键点在代码注释里面都有提及,重要的地方再强调一下:

    • 如果要使用代理方法,需要设置代理,但从NSURLSession的头文件发现session的delegate属性是只读的.因此设置代理要通过session的初始化方法赋值:sessionWithConfiguration:delegate:delegateQueue:其中:
      • configuration参数(文章开始提到的)需要传递一个配置,我们暂且使用默认的配置[NSURLSessionConfiguration defaultSessionConfiguration]就好(后面会说下这个配置是干嘛用的);
      • delegateQueue参数表示协议方法将会在哪个队列(NSOperationQueue)里面执行.
    • NSURLSession在接收到响应的时候要先对响应做允许处理:completionHandler(NSURLSessionResponseAllow);,才会继续接收服务器返回的数据,进入后面的代理方法.值得一提的是,如果在接收响应的时候需要对返回的参数进行处理(如获取响应头信息等),那么这些处理应该放在前面允许操作的前面.

    NSURLSessionDownloadTask

    文件下载可以使用NSURLSessionDownloadTask这个子类.

    简单下载

    NSURLSessionDownloadTask同样提供了通过NSURL和NSURLRequest两种方式来初始化并通过block进行回调的方法.下面以NSURL初始化为例:

    SURLSession *session = [NSURLSession sharedSession];
    NSURL *url = [NSURL URLWithString:@"http://www.daka.com/resources/image/icon.png"] ;
    NSURLSessionDownloadTask *task = [session downloadTaskWithURL:url completionHandler:^(NSURL *location, NSURLResponse *response, NSError *error) {
        // location是沙盒中tmp文件夹下的一个临时url,文件下载后会存到这个位置,由于tmp中的文件随时可能被删除,所以我们需要自己需要把下载的文件挪到需要的地方
        NSString *path = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:response.suggestedFilename];
        // 剪切文件
        [[NSFileManager defaultManager] moveItemAtURL:location toURL:[NSURL fileURLWithPath:path] error:nil];
    }];
        // 启动任务
        [task resume];
    

    Tips:

    • 需要注意的就是需要将下载到tmp文件夹的文件转移到需要的目录.原因在代码中已经贴出.
    • response.suggestedFilename是从相应中取出文件在服务器上存储路径的最后部分,如数据在服务器的url为http://www.daka.com/resources/image/icon.png, 那么其suggestedFilename就是icon.png.

    NSURLSessionDownloadDelegate代理方法

    同样的,downloadTask也提供了配套的代理方法

    // 每次写入调用(会调用多次)
    - (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didWriteData:(int64_t)bytesWritten totalBytesWritten:(int64_t)totalBytesWritten totalBytesExpectedToWrite:(int64_t)totalBytesExpectedToWrite {
    // 可在这里通过已写入的长度和总长度算出下载进度
    CGFloat progress = 1.0 * totalBytesWritten / totalBytesExpectedToWrite; NSLog(@"%f",progress);
    }
    
    // 下载完成调用
    - (void)URLSession:(NSURLSession *)session
          downloadTask:(NSURLSessionDownloadTask *)downloadTask
          didFinishDownloadingToURL:(NSURL *)location {
        // location还是一个临时路径,需要自己挪到需要的路径(caches下面)
        NSString *filePath = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:downloadTask.response.suggestedFilename];
        [[NSFileManager defaultManager] moveItemAtURL:location toURL:[NSURL fileURLWithPath:filePath] error:nil];
    }
    
    // 任务完成调用
    - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
    
    }
    

    NSURLSessionUploadTask

    在NSURLSession中,文件上传方式主要有以下两种:

    NSURLSessionUploadTask *task =
    [[NSURLSession sharedSession] uploadTaskWithRequest:request
                                               fromFile:fileName
                                      completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
    }];
    

     [self.session uploadTaskWithRequest:request
                                fromData:body
                       completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
     NSLog(@"-------%@", [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil]);
     }];
    

    处于安全性考虑,通常我们会使用POST方式进行文件上传,所以较多使用第二种方式.

    但是,NSURLSession并没有为我们提供比NSURLConnection更方便的文件上传方式.方法中body处的参数需要填写request的请求体(http协议规定格式的大长串).因为你有90%的可能性用了AFNetworking,即使是自己写的应该也是copy,所以代码就不贴了我们只说方法呵呵哒.

    断点下载

    NSURLSessionDownloadTask提供了与断点下载相关的几个方法:

    // 使用这种方式取消下载可以得到将来用来恢复的数据,保存起来
    [self.task cancelByProducingResumeData:^(NSData *resumeData) {
        self.resumeData = resumeData;
    }];
    
    // 由于下载失败导致的下载中断会进入此协议方法,也可以得到用来恢复的数据
    - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
    {
        // 保存恢复数据
        self.resumeData = error.userInfo[NSURLSessionDownloadTaskResumeData];
    }
    
    // 恢复下载时接过保存的恢复数据
    self.task = [self.session downloadTaskWithResumeData:self.resumeData];
    // 启动任务
    [self.task resume];
    

    以目前我对NSURLSession的理解这种断点下载只支持应用内断点,如果程序在下载过程中途关闭,则不能恢复下载.(暂时对NSURLSession理解还不全面,不敢妄下断论,如有不妥简友们可以沟通下)

    其他

    此外,task们自身有都拥有下面几个方法

    - (void)suspend;
    - (void)resume;
    - (void)cancel;
    

    suspend可以让当前的任务暂停

    resume方法不仅可以启动任务,还可以唤醒suspend状态的任务

    cancel方法可以取消当前的任务,你也可以向处于suspend状态的任务发送cancel消息,任务如果被取消便不能再恢复到之前的状态.

    NSURLSessionConfiguration

    简单地说,就是session的配置信息.如:

    NSURLSessionConfiguration *config = [NSURLSessionConfiguration defaultSessionConfiguration];
    // 超时时间
    config.timeoutIntervalForRequest = 10;
    // 是否允许使用蜂窝网络(后台传输不适用)
    config.allowsCellularAccess = YES;
    // 还有很多可以设置的属性
    

    有没有发现我们使用的Configuration都是默认配置:[NSURLSessionConfiguration defaultSessionConfiguration],其实它的配置有三种类型:

    + (NSURLSessionConfiguration *)defaultSessionConfiguration;
    + (NSURLSessionConfiguration *)ephemeralSessionConfiguration;
    + (NSURLSessionConfiguration *)backgroundSessionConfigurationWithIdentifier:(NSString *)identifier
    

    表示了NSURLSession几种不同的工作模式.
    默认的配置会将缓存存储在磁盘上,第二种瞬时会话模式不会创建持久性存储的缓存,第三种后台会话模式允许程序在后台进行上传下载工作.

    除了支持任务的暂停和断点续传,我觉得NSURLSession之于NSURLConnection的最伟大的进步就是支持后台上传下载任务,这又是一个可以深入讨论的话题.但在这方面我还没有进行深入的研究,待后续了解之后另行开贴.

    PS:AFNetWorking从2.0版本就有了基于NSURLSession的系列封装,感兴趣的童鞋自行前往了解.

    相关文章

      网友评论

      • 程序H:来挖坟了。。。。

        如设置completionHandler: { (<#Data?#>, <#URLResponse?#>, <#Error?#>)代理方法不会执行。

        引用dataTask时方法不引用block函数,代理方法才会执行。
        let task = session.dataTask(with: request)
        task.resume()
      • PGOne爱吃饺子:大哥 江湖救急啊
      • 金鲤:有的程序员老了 ,还没听过NSURLSession
        有的程序员还嫩 ,没用过NSURLConnection
        有的程序员很单纯 ,他只知道AFN
        写一首比较好一点的诗^_^
        金鲤:……,本来控制后半句的逗号都对齐的,:sweat:
      • tangbin583085:挺不错,上班地铁上看的,很有价值
      • macfai:请问楼主,https在nsurlsession中怎么配置
      • 44d3387e09f3:要使用FTP 应该怎么用这个类呢
      • LD_左岸:- (void) download{
        NSURLSessionConfiguration *defaultConfigObject = [NSURLSessionConfiguration defaultSessionConfiguration];
        NSURLSession *defaultSession = [NSURLSession sessionWithConfiguration: defaultConfigObject delegate:self delegateQueue: nil]; /*[NSOperationQueue mainQueue]*/

        NSURLSessionDownloadTask * downloadTask1 =[ defaultSession downloadTaskWithURL:url];
        [downloadTask1 resume];
        }

        - (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didFinishDownloadingToURL:(NSURL *)location
        {
        移动文件从tmp目录到自己的目录
        }

        有个错误:

        __NSCFLocalDownloadFile: error 24 opening for ioChannel, file: /private/var/mobile/Containers/Data/Application/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/tmp/CFNetworkDownload_pMZR5D.tmp
        这个报错在Debug区大量被打印 大神可知咋回事
      • ebay_Happy:感谢大神分享
      • FindCrt:好诗好诗~
      • 山林间迷雾能不能当障眼法的内容:上面 简单GET请求 里面

        // 通过URL初始化task,在block内部可以直接对返回的数据进行处理
        NSURLSessionTask *task = [session dataTaskWithURL:url
        completionHandler:
        怎么用的 NSURLSessionTask *task ???????? 他不是一个抽象类吗 不是要用他的子类吗 怎么直接用NSURLSessionTask ???
      • ImmortalSummer:看完了,感觉帮助很大,比网上很多看似专业其实东拼西凑的东西更容易理解,在此谢过
      • block_smile:"以目前我对NSURLSession的理解这种断点下载只支持应用内断点,如果程序在下载过程中途关闭,则不能恢复下载.(暂时对NSURLSession理解还不全面,不敢妄下断论,如有不妥简友们可以沟通下)"我觉得可以啊, 程序关闭时,可以利用断点续传的resumeData写入沙河,重启后可判断沙河是否有文件,接着断点续传
        JoinPerson:@block_smile 关键是在 Kill 程序的一瞬间你如何获得 resumeData
        block_smile:@JoinPerson 你可以通过resumeData writetofile方法写入沙盒,开启之后再取到这个判断,创建续传任务继续写入。
        JoinPerson:在kill程序的一瞬间你怎么获得此时的 resumeData ?
      • b9ea8ac47824:断点下载那里讲到说应用内可断点,能否使用backgroundSessionConfiguration来请求下载,这个session下载完毕后,系统会在任务完成或者证书请求的时候在后天唤醒app,相应调用UIApplication的相关方法,使用之前配置session时候的identifier来创建新的session并在mainquene中启动,以便系统知道session完成suspend app,seesion完成后,调用代理方法完成操作。
      • 魔性佛心:简单下载那边 NSURLSession 写错了,写成了SURLSession
      • catcherdream:写的不详细,这方面一点不了解的不适合看。
      • 皇垚:resumeData 数据持久化就可以实现退出,重登录的继续下载
        北雪未央吾思卿如狂:直接杀死程序是得不到resumeData数据的
      • 元哥830:有帮助
      • bd8b3a8e64a7:谢谢!
      • cc130922a6b4:请问在改下载文件的路径时location要咋用?按照你上面的方法 我老是崩溃 说是URL resource can't be nil 要怎么解决
      • 陈大帅:很不错,学习了
      • xfx_itachi:简单get请求那里,completionHandler:^(NSData *data, NSURLResponse *response, NSError error)
        error 前面少了指针
      • 28fd3cea58da:我想问下你是怎么处理重定向的请求的,我重定向之后页面的url并没有改变。
      • 前进的火车2015:很棒的一遍技术文章,谢谢分享
      • Wang66:生成NSURLSessionDataTask实例后,是不是不要设置self为其代理,就可以执行代理方法???我看了接口文档,确实没有delegate属性或者相关方法,感觉有些怪异啊? :sweat:
        Wang66:@Wang66 额,我查到了答案... session和task都公用一个,只要给session设置了代理,就代表也是task的代理... :sweat:
      • 403065e6418f:好使好事啊,谢谢大神了
      • b8e0c9b678f1:好诗好诗
      • smallLabel:看完了,写的挺好的,马了
      • cyhai:写的不错,就是感觉有些粗心
      • 全民同學:"拢共"这个成用的好啊,生动形象
      • 2e0cde7b84aa:内容很完善了,学习了。但是代码有明显的错误,缺少了指针符合“*”,如下面的代码(NSURLSession )session ----> (NSURLSession *)session

        // 1.接收到服务器的响应
        - (void)URLSession:(NSURLSession )session dataTask:(NSURLSessionDataTask )dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler {
        // 允许处理服务器的响应,才会继续接收服务器返回的数据
        completionHandler(NSURLSessionResponseAllow);
        }

        // 2.接收到服务器的数据(可能调用多次)
        - (void)URLSession:(NSURLSession )session dataTask:(NSURLSessionDataTask )dataTask didReceiveData:(NSData *)data {
        // 处理每次接收的数据
        }

        // 3.请求成功或者失败(如果失败,error有值)
        - (void)URLSession:(NSURLSession )session task:(NSURLSessionTask )task didCompleteWithError:(NSError *)error {
        // 请求完成,成功或者失败的处理
        }
        程序H:吹毛求疵
        CoderAO:@Oye0815 谢谢提醒,已经改过来了,之前都是在其他编辑器编辑然后复制过来,不知道复制过来的时候为什么有很多*就没掉了...这些代码都是Xcode里面敲的然后复制过去,理论上不会出现缺少*的情况.
      • 扣肉快快跑:省略的上传还要查看别资料了
      • NiuBaBa:mark 请问代码的排版是怎么弄的?
        NiuBaBa:@annj 谢谢了
        antns丶:@NiuBaBa 简书设置里选择markDown编辑器,<pre>中间放代码</pre>
      • iOS_愛OS:漂亮
      • 小凡凡520:good mark

      本文标题:使用NSURLSession

      本文链接:https://www.haomeiwen.com/subject/nblpqttx.html