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