大文件下载注意事项
- 若不对下载的文件进行转存,会造成内存消耗急剧升高,甚至耗尽内存资源,造成程序终止。
- 在文件下载过程中通常会出现中途停止的状况,若不做处理,就要重新开始下载,浪费流量。
大文件下载的解决方案
-
对下载文件进行处理,每下载一点数据,就将数据写到磁盘中(通常是沙盒中),避免在内存累积数据(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];
-