使用NSURLSessionDownloadTask来下载任务(可以实现断点下载,离线下载还是使用NSURLSessionDataTask)
NSURLSessionDownloadTask断点下载 遵守NSURLSessionDownloadDelegate
#pragma mark 断点下载 不需要把每段下载的数据保存到本地 只需要把下载完成的数据保存就可以了 断点下载是通过resumeData(cancelByProducingResumeData保存下载到哪里了)然后downloadTaskWithResumeData继续从上次保存的位置开始下载 app在后台下载和app没关系(系统调度的另一个进程在进行后台下载)
-(void)dataDownLoadTask:(UIButton*)sender{
// 按钮状态取反
sender.selected= !sender.isSelected;
if (nil == self.downloadTask) { // [开始下载/继续下载]
if (self.resumeData) { // [继续下载]
// 传入上次暂停下载返回的数据,就可以恢复下载
self.downloadTask = [self.session downloadTaskWithResumeData:self.resumeData];
// 开始任务
[self.downloadTask resume];
self.resumeData = nil;
}else{ // [开始下载]:从0开始下载
NSURL* url = [NSURL URLWithString:@"http://dldir1.qq.com/qqfile/QQforMac/QQ_V5.4.0.dmg"];
// 创建任务
self.downloadTask = [self.session downloadTaskWithURL:url];
// 开始任务
[self.downloadTask resume];
}
}else{ // [暂停下载]
__weaktypeof(self) weakSelf =self;
[self.downloadTask cancelByProducingResumeData:^(NSData *resumeData) {
// resumeData:包含了继续下载的位置\下载的路径
weakSelf.resumeData= resumeData;
weakSelf.downloadTask=nil;
}];
}
}
代理回调方法
/**
* 每次写入数据到临时文件时,就会调用一次这个方法。可在这里获得下载进度
*
* @parambytesWritten 这次写入的文件大小
* @paramtotalBytesWritten 已经写入沙盒的文件大小
* @paramtotalBytesExpectedToWrite 文件总大小
*/
- (void)URLSession:(NSURLSession*)session downloadTask:(NSURLSessionDownloadTask*)downloadTask
didWriteData:(int64_t)bytesWritten
totalBytesWritten:(int64_t)totalBytesWritten
totalBytesExpectedToWrite:(int64_t)totalBytesExpectedToWrite
{
NSLog(@"读取字节大小%lld,当前线程%@,%@",bytesWritten,[NSThread currentThread],NSHomeDirectory());
// 下载进度
dispatch_async(dispatch_get_main_queue(), ^{
self.progress.progress=1.0* totalBytesWritten / totalBytesExpectedToWrite;
self.progressLabel.text= [NSStringstringWithFormat:@"当前下载进度:%.2f%%",100.0* totalBytesWritten / totalBytesExpectedToWrite];
});
}
/**
* 恢复下载后调用 不常在这里面操作
*/
- (void)URLSession:(NSURLSession*)session downloadTask:(NSURLSessionDownloadTask*)downloadTask
didResumeAtOffset:(int64_t)fileOffset
expectedTotalBytes:(int64_t)expectedTotalBytes
{
}
/**
* 文件下载完毕时调用
*/
- (void)URLSession:(NSURLSession*)session downloadTask:(NSURLSessionDownloadTask*)downloadTask
didFinishDownloadingToURL:(NSURL*)location
{
NSLog(@"当前线程%@",[NSThread currentThread]);
// 文件将要移动到的指定目录
NSString *documentsPath = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) firstObject];
// 新文件路径
NSString *newFilePath = [documentsPath stringByAppendingPathComponent:@"QQ_V5.4.0.dmg"];
NSLog(@"File downloaded to: %@",newFilePath);
// 移动文件到新路径
[[NSFileManager defaultManager] moveItemAtPath:location.path toPath:newFilePath error:nil];
}
这样就可以了
NSURLSessionDataTask离线下载 遵守NSURLSessionDataDelegate
/**
* offlineDownLoadTask的懒加载,这里设置请求头中的Range
*/
- (NSURLSessionDataTask*)offlineDownLoadTask {
if (!_offlineDownLoadTask) {
// 创建下载URL
NSURL *url = [NSURL URLWithString:@"http://dldir1.qq.com/qqfile/QQforMac/QQ_V5.4.0.dmg"];
// 2.创建request请求
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
// 设置HTTP请求头中的Range 实现断点下载的关键
NSString *range = [NSString stringWithFormat:@"bytes=%zd-", self.currentLength];
[requestsetValue:range forHTTPHeaderField:@"Range"];
// 3. 下载
_offlineDownLoadTask = [self.session dataTaskWithRequest:request];
}
return _offlineDownLoadTask;
}
//离线下载 离线下载需要把每段下载的数据保存到本地 这样filehandler就可以操作文件的开始读写的位置 实现的核心部分是 [NSFileHandle fileHandleForWritingAtPath:path]对每次接受到的数据写入到沙盒 [self.fileHandle seekToEndOfFile] [self.fileHandle writeData:data]移动指针到文件结尾 然后在文件结尾继续拼接数据 这样就实现了重启app 后台退出app能够继续从沙盒中的位置开始下载(也叫离线下载) app在后台下载和app没关系(系统调度的另一个进程在进行后台下载)
-(void)dataDownLoadTask:(UIButton*)sender{
// 按钮状态取反
sender.selected= !sender.isSelected;
if (sender.selected) { // [开始下载/继续下载]
// 沙盒文件路径
NSString *path = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:@"QQ_V5.4.0.dmg"];
NSIntegercurrentLength = [selffileLengthForPath:path];
if(currentLength >0) { // [继续下载]
self.currentLength= currentLength;
}
[self.offlineDownLoadTask resume];
}else{
//暂停
// [self.offlineDownLoadTask suspend];
[self.offlineDownLoadTask cancel];
self.offlineDownLoadTask = nil;
}
}
/**
* 获取已下载的文件大小
*/
- (NSInteger)fileLengthForPath:(NSString*)path {
NSIntegerfileLength =0;
NSFileManager *fileManager = [NSFileManager defaultManager]; // default is not thread safe
if([fileManagerfileExistsAtPath:path]) {
NSError*error =nil;
NSDictionary*fileDict = [fileManagerattributesOfItemAtPath:patherror:&error];
if(!error && fileDict) {
fileLength = [fileDictfileSize];
}
}
returnfileLength;
}
代理回调
/**
* 接收到响应的时候:创建一个空的沙盒文件
*/
- (void)URLSession:(NSURLSession*)session dataTask:(NSURLSessionDataTask*)dataTask didReceiveResponse:(NSURLResponse*)response completionHandler:(void(^)(NSURLSessionResponseDisposition))completionHandler
{
NSLog(@"下载长度%lld",response.expectedContentLength);
// 获得下载文件的总长度:请求下载的文件长度 + 当前已经下载的文件长度
self.fileLength = response.expectedContentLength + self.currentLength;
// 沙盒文件路径
NSString *path = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:@"QQ_V5.4.0.dmg"];
NSLog(@"File downloaded to: %@",path);
// 创建一个空的文件到沙盒中
NSFileManager *manager = [NSFileManager defaultManager];
if(![managerfileExistsAtPath:path]) {
// 如果没有下载文件的话,就创建一个文件。如果有下载文件的话,则不用重新创建(不然会覆盖掉之前的文件)
[managercreateFileAtPath:path contents:nil attributes:nil];
}
// 创建文件句柄
self.fileHandle = [NSFileHandle fileHandleForWritingAtPath:path];
// 允许处理服务器的响应,才会继续接收服务器返回的数据
completionHandler(NSURLSessionResponseAllow);
}
/**
* 接收到具体数据:把数据写入沙盒文件中
*/
- (void)URLSession:(NSURLSession*)session dataTask:(NSURLSessionDataTask*)dataTask didReceiveData:(NSData*)data
{
NSLog(@"当前线程%@",[NSThread currentThread]);
// 指定数据的写入位置 -- 文件内容的最后面
[self.fileHandle seekToEndOfFile];
// 向沙盒写入数据
[self.fileHandle writeData:data];
// 拼接文件总长度
self.currentLength += data.length;
NSLog(@"%ld",self.currentLength);
// 下载进度
self.progress.progress= 1.0 * self.currentLength / self.fileLength;
self.progressLabel.text = [NSString stringWithFormat:@"当前下载进度:%.2f%%",100.0 * self.currentLength / self.fileLength];
}
/**
* 下载完文件之后调用:关闭文件、清空长度 (暂停suspend操作不会走该方法,取消cancel操作会走该方法)
*/
- (void)URLSession:(NSURLSession*)session task:(NSURLSessionTask*)task didCompleteWithError:(NSError*)error
{
// 关闭fileHandle
[self.fileHandle closeFile];
self.fileHandle = nil;
// 清空长度
self.currentLength = 0;
self.fileLength = 0;
}
个人github地址:zhuweigea (LaoZhu) · GitHub