关于文件下载

作者: Dylan_Yu | 来源:发表于2016-05-05 21:57 被阅读1565次

    使用xcode自带的代理方法。

    代理方法:
    //这个是开始下载时 调用的方法。
    - (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
    //这个也是每次都在调用的方法。
    data(代表每次下载的数据大小)
    - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
    
    //这个方法是在下载时一直会调用的。这里有三个参数。
    bytesWritten(代表本次下载下载了多少)
    totalBytesWritten(代表已经下载了多少)
    totalBytesExpectedToWrite(代表文件总大小是多少)
    - (void)connection:(NSURLConnection *)connection   didSendBodyData:(NSInteger)bytesWritten
                                                     totalBytesWritten:
    (NSInteger)totalBytesWritten
                                             totalBytesExpectedToWrite:(NSInteger)totalBytesExpectedToWrite;
    
    //这个方法在下载完成后 会自动调用。
    connection(这个参数是下完完成后 文件的路径)
    - (void)connectionDidFinishLoading:(NSURLConnection *)connection;
    
    

    1.使用NSUrlContionDownloadDelegate.

    1)内存不会暴涨 但是找不到下载完毕之后的文件。
    2)可以直接显示下载进度。

    2.使用NSUrlconnectionDataDelegate.

    直接设置NSMutableData属性来接受下载完毕的数据 .
    1)内存依然“暴涨”:相当于还是先将整个文件下载到内存中 然后在写入沙盒中

    1. 无法直接显示下载进度。需要手动计算。

    3.利用NSfileHanddle

    • NSFileHandle :文件操作句柄,用来拼接文件。
    //用已下载的本地路径去创建handle
     NSFileHandle * handle = [NSFileHandle fileHandleForWritingAtPath:@"/Users/ym/Desktop/haha.zip"];
        //如果handle创建成功 代表本地路径有文件。
        if (handle) {
            操作句柄到最后
            [handle seekToEndOfFile];
            
            // 拼接文件/写入文件
            [handle writeData:data];
            
            // 关闭句柄
            [handle closeFile];
           
        }else{
        //本地路径没有文件。
    就重新创建一个本地文件路径
            [data writeToFile:@"/Users/ym/Desktop/haha.zip" atomically:YES];
    
        }
    

    问题:多次下载,会直接在之前的文件后拼接文件—>得不到正确的文件数据。

    4. NSOutputStream

    //首先要根据文件路径来创建stream 。如果你的这个路径没有文件。他会自动给你创建一个文件。

       NSOutputStream * stream = [NSOutputStream outputStreamToFileAtPath:@" /Users/ym/Desktop/haha.zip" append:YES];
    //创建一个属性。来存储文件和关闭stream
        self.stream = stream;
    //开启任务。
        [self.stream open];
    
    //在方法里 来拼接data内容。
    - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data{
        [self.stream write:data.bytes maxLength:data.length];
    }
    
    - (void)connectionDidFinishLoading:(NSURLConnection *)connection{
    //在下载结束任务后,关闭stream任务。
        [self.stream close];
    
    }
    

    这样的下载任务。还是会有和handle一样的问题。多次下载和暂停下载之后继续下载。拼接的文件都是错误的。

    所以这里就引入了断点续传功能。

    用HEAD请求来获取下载文件的总大小

    • HTTP协议方法: HEAD 请求
    • HEAD 请求: 只获取服务器的响应头信息,不获取文件内容(实体内容).
    • 一般在文件下载的时候,提前获取文件信息.
    • HEAD请求: 和 GET 请求是一样的.只是不获取文件内容(实体内容).
    //文件路径
    NSString *urlString = @"http://127.0.0.1/music.zip";
        
        NSURL *url = [NSURL URLWithString:urlString];
        //创建请求
        NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
        
        // 设置请求方法:
        request.HTTPMethod = @"HEAD";
        
        //  发送 HEAD 请求.
        // HEAD 请求使用什么方法发送? ---- 一般使用同步方式发送HEAD请求.
        
     NSURLResponse *response = nil;
    [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:NULL];
        //打印出来的就是你的所需要的文件总大小。
        NSLog(@"response:%zd",response.expectedContentLength);
    

    获取本地路径中 文件的大小。

    //这里返回值是一个字典。
     [[NSFileManager defaultManager] attributesOfItemAtPath:self.filePath error:NULL];
    
    //这个就是打印字典的信息。
    我们所需要的就是NSFileSize
    **  NSFileCreationDate = "2016-05-05 12:59:14 +0000";
    **    NSFileExtensionHidden = 0;
    **    NSFileGroupOwnerAccountID = 20;
    **    NSFileGroupOwnerAccountName = staff;
    **    NSFileModificationDate = "2016-05-05 12:59:17 +0000";
    **    NSFileOwnerAccountID = 501;
    **    NSFilePosixPermissions = 420;
    **    NSFileReferenceCount = 1;
    **    NSFileSize = 573278888;
    **    NSFileSystemFileNumber = 19578034;
    **    NSFileSystemNumber = 16777217;
    **    NSFileType = NSFileTypeRegular;
    

    所以要接收这个字典 同时返回NSFileSize.获取本地文件已经下载的大小。

    知道了已经下载的大小,还有总大小。我们就可以根据这两个数值来进行断点续传了。

    断点续传需要的请求---Range请求。

    设置请求头信息。
    <1>格式:
    Range格式:
    bytes = x-y 从x位置开始下载,下载y个字节。
    bytes = x- 从x开始 下载到完毕。
    bytes = -x 从开始下载x个字节。
    <2>特点:
    一旦Range 属性设置成功,相应的状态码就是206

    //startSize 就是上文提到的dict中的NSFileSize。要把这个oc属性转化为字符串。
    //bytes=zd - 就是从你断点的位置下载到文件结束
    
      NSString *range = [NSString stringWithFormat:@"bytes=%zd-",startSize];
       [request setValue:range forHTTPHeaderField:@"Range"];
    

    下面讲一种NSURLSession的下载

    • 利用代理NSURLSessionDownloadDelegate
      1.主要利用的代理方法。
    //这个是每次下载都调用的方法。和上边的一样。
    -(void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
          didWriteData:(int64_t)bytesWritten
     totalBytesWritten:(int64_t)totalBytesWritten
    totalBytesExpectedToWrite:(int64_t)totalBytesExpectedToWrite
    
    //下载完成之后调用的方法。
    -(void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
    didFinishDownloadingToURL:(NSURL *)location
    
    //取消任务。
    -(NSURLSessionDownloadTask *)downloadTaskWithResumeData:(NSData *)resumeData
    
    • 断点续传请求。调用上面的方法。
        NSString * urlString = @"http://127.0.0.1/music.zip";
        NSURL * url  = [NSURL URLWithString:urlString];
        //创建可变请求。添加代理。
        NSURLSession * session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:nil];
    //创建下载任务。
        NSURLSessionDownloadTask * task = [session downloadTaskWithURL:url];
        self.oldSession = session;
        self.task = task;
        [task resume];
    
    • 暂停下载
    [self.task cancelByProducingResumeData:^(NSData * _Nullable resumeData) {
        self.resumeData = resumeData;
        
    }];
      self.task = nil;
    
    
    • 继续下载
    if (self.resumeData) {
            self.task = [self.oldSession downloadTaskWithResumeData:self.resumeData];
            [self.task resume];
        }
    

    相关文章

      网友评论

      • 90后的晨仔:您好,cancelByProducingResumeData:^(NSData * _Nullable resumeData{} 中resumeData为nil,ios 11.4请问怎么解决?
      • 魔魔571:大佬,问一下,(NSData * _Nullable resumeData) 代码块里,这个 resumeData 为空,是什么情况,有遇到过吗
        谎言的背后:@魔魔571 if (self.resumeData) {
        if (IS_IOS10ORLATER) {
        self.downloadTask = [self.backgroundSession downloadTaskWithCorrectResumeData:self.resumeData];
        } else {
        self.downloadTask = [self.backgroundSession downloadTaskWithResumeData:self.resumeData];
        }
        [self.downloadTask resume];
        self.resumeData = nil;
        }
        魔魔571:@谎言的背后 是的
        谎言的背后:是不是ios10
      • Colleny_Z:楼主,我问个问题:假如我下载一个文件,每一次在didReceiveData后,用NSfileHanddle在文件尾部追加字节data,那么问题来了,会不会遇到这样的情况:比如我在文件尾部本应追加100个字节,但刚写入50个字节,第二次又接受到了100个字节,但是NSfileHanddle寻找尾部是从上次的50开始,这样就会导致后面字节都错位了,这个问题有可能么,如何避免?
        Dylan_Yu:如果你想每次都在尾部追加下载的文件的话 就采用断点续传那种操作。如果下载的时候加一个判断呗。如果下载相同东西的话 就不重置下载操作 return就好了。如果是新的下载操作。如果同时下载多个操作的话 你就可以加判断存进数组里 之后顺序追加就好了。
      • qichen:请问一下,NSURLSession如何获取服务器下载列表
        qichen:@Richard_Yu 比如说我想下载a.xml这样一个文件,我想先获取a.xml这个文件的文件夹下的所有文件的列表。
        Dylan_Yu:什么叫服务器下载列表。是你下载的内容 信息 还是说 服务器返回给你的信息?
      • Mr吴标:学习中
        Dylan_Yu:@Mr吴标 :smile: 共同学习,共同进步

      本文标题:关于文件下载

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