OC - 16.大文件下载

大文件下载注意事项


  • 若不对下载的文件进行转存,会造成内存消耗急剧升高,甚至耗尽内存资源,造成程序终止。
  • 在文件下载过程中通常会出现中途停止的状况,若不做处理,就要重新开始下载,浪费流量。

大文件下载的解决方案


  • 对下载文件进行处理,每下载一点数据,就将数据写到磁盘中(通常是沙盒中),避免在内存累积数据(NSURLConnection下载)

    • 使用NSFileHandle类实现写数据
    • 使用NSOutputStream类实现写数据
  • 当下载任务终止时,记录任务终止时的位置信息,以便下次开始继续下载

大文件下载(NSURLConnection)


  • 未支持断点下载
  • 使用NSURLConnection的代理方式下载文件
  • 在下载任务的不同阶段回调的代理方法中,完成转移下载文件,及记录终止位置的任务
  • 使用NSFileHandle类实现写数据的下载步骤(完整核心代码)

    • 设置相关成员属性

      /**所要下载文件的总长度*/
      @property (nonatomic, assign) NSInteger contentLength;
      /**已下载文件的总长度*/
      @property (nonatomic, assign) NSInteger currentLength
      /**文件句柄,用来实现文件存储*/
      @property (nonatomic, strong) NSFileHandle *handle;
    • 创建、发送请求

      // 1. 创建请求路径
      NSURL *url = [NSURL URLWithString:@"此处为URL字符串"];
      // 2. 将URL封装成请求
      NSURLRequest *request = [NSURLRequest requestWithURL:url];
      // 3. 通过NSURLConnection,并设置代理
      [NSURLConnection connectionWithRequest:request delegate:self];
    • 遵守代理协议NSURLConnectionDataDelegate,实现代理方法

      /**
      * 接收到服务器响应时调用的方法
      */
      - (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSHTTPURLResponse *)response
      {
      	//获取所要下载文件的总长度
      	self.contentLength = [response.allHeaderFields[@"Content-Length"] integerValue];
      	//拼接一个沙盒中的文件路径
      	NSString *filePath = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:@"minion_15.mp4"];
      	//创建指定路径的文件
      	[[NSFileManager defaultManager] createFileAtPath:filePath contents:nil attributes:nil];
      	//创建文件句柄
      	self.handle = [NSFileHandle fileHandleForWritingAtPath:filePath];
      }
      /**
      * 接收到服务器的数据时调用的方法
      */
      - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
      {
      	//定位到文件尾部,将服务器每次返回的文件数据都拼接到文件尾部
      	[self.handle seekToEndOfFile];
      	//通过文件句柄,将文件写入到沙盒中
      	[self.handle writeData:data];
      	//拼接已下载文件总长度
      	self.currentLength += data.length;
      	//计算下载进度
      	CGFloat progress = 1.0 * self.currentLength / self.contentLength;
      }
      /**
      * 文件下载完毕时调用的方法
      */
      - (void)connectionDidFinishLoading:(NSURLConnection *)connection
      {
      	//关闭文件句柄,并清除
      	[self.handle closeFile];
      	self.handle = nil;
      	//清空已下载文件长度
      	self.currentLength = 0;
      }
  • 使用NSOutputStream类实现写数据的下载步骤(部分代码,其他部分代码同上

    • 设置NSOutputStream成员属性

      @property (nonatomic, strong) NSOutputStream *stream;
    • 初始化NSOutputStream对象,打开输出流

      /**接收到服务器响应的时候调用*/
      - (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
      {
      	//获取下载数据保存的路径
      	NSString *cache = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject];
      	NSString *filePath = [cache stringByAppendingPathComponent:response.suggestedFilename];
      	//利用NSOutputStream往filePath文件中写数据,若append参数为yes,则会写到文件尾部
      	self.stream = [[NSOutputStream alloc] initToFileAtPath:filePath append:YES];
      	//打开数据流
      	[self.stream open];
      }
    • 写文件数据

      /**接收到数据的时候调用*/
      - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
      {
      	[self.stream write:[data bytes] maxLength:data.length];
      }
    • 关闭输出流

      /**数据下载完毕的时候调用*/
      - (void)connectionDidFinishLoading:(NSURLConnection *)connection
      {
      	[self.stream close];
      }

大文件下载(NSURLSession)


  • 支持断点下载,自动记录停止下载时断点的位置
  • 遵守NSURLSessionDownloadDelegate协议
  • 使用NSURLSession下载大文件,被下载文件会被自动写入沙盒的临时文件夹tmp中
  • 下载完毕,通常需要将已下载文件移动其他位置(tmp文件夹中的数据被定时删除),通常是cache文件夹中
  • 详细的下载步骤

    • 设置下载任务task的为成员变量

      @property (nonatomic, strong) NSURLSessionDownloadTask *task;
    • 获取NSURLSession对象

      NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[[NSOperationQueue alloc] init]];
    • 初始化下载任务任务

      self.task = [session downloadTaskWithURL:(此处为下载文件路径URL)];
    • 实现代理方法

      /**每当写入数据到临时文件的时候,就会调用一次该方法,通常在该方法中获取下载进度*/
      - (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;
      }
      
      /**任务终止时调用的方法,通常用于断点下载*/
      - (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didResumeAtOffset:(int64_t)fileOffset expectedTotalBytes:(int64_t)expectedTotalBytes
      {
      	//fileOffset:下载任务中止时的偏移量
      }
      
      /**遇到错误的时候调用,error参数只能传递客户端的错误*/
      - (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
      { }
      
      /**下载完成的时候调用,需要将文件剪切到可以长期保存的文件夹中*/
      - (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didFinishDownloadingToURL:(NSURL *)location
      {
      	//生成文件长期保存的路径
      	NSString *file = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:downloadTask.response.suggestedFilename];
      	//获取文件句柄
      	NSFileManager *fileManager = [NSFileManager defaultManager];
      	//通过文件句柄,将文件剪切到文件长期保存的路径
      	[fileManager moveItemAtURL:location toURL:[NSURL fileURLWithPath:file] error:nil];
      }
    • 操作任务状态

      /**开始/继续下载任务*/
      [self.task resume];
      
      /**暂停下载任务*/
      [self.task suspend];

       

 
 
posted @ 2015-11-07 20:14 空山新雨funny 阅读(...) 评论(...) 编辑 收藏