1 .为能在后台继续下载,此处的NSURLSessionConfiguration需置为backgroundSessionConfigurationWithIdentifier。为能在关闭应用后继续恢复下载任务,此处需对其设置identifier。
- (NSURLSession *)backgroundURLSession {
static NSURLSession *session = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
NSString *identifier = @"com.yourcompany.appId.BackgroundSession";
NSURLSessionConfiguration* sessionConfig = [
backgroundSessionConfigurationWithIdentifier:identifier];
session = [NSURLSession sessionWithConfiguration:sessionConfig
delegate:self
delegateQueue:[NSOperationQueue mainQueue]];
});
return session;
}
2. 对于应用程序退出的下载:
当应用程序时强制关闭时,若下载任务没有完成,则退出后不会继续下载;当应用程序因为崩溃等原因退出时,为完成的下载任务仍会继续下载直至完成。
前台下载情况:
一. NSURLSession前台下载开始------------>下载结束:
1.在下载过程中,NSURLSessionDownloadDelegate的
URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didWriteData:(int64_t)bytesWritten totalBytesWritten:(int64_t)totalBytesWritten totalBytesExpectedToWrite (int64_t)totalBytesExpectedToWrite
接口会多次调用,在此接口中可进行下载进度的记录或更新UI的下载进度条。
2.当下载顺利结束时,NSURLSessionDownloadDelegate的
URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
didFinishDownloadingToURL:(NSURL *)location
接口会被调用一次,此接口中的location参数记录下载的tmp文件的存储路径,在此接口中需将下载的tmp文件重新保存至最终的保存地址中。
3.若下载成功或失败时,NSURLSessionTaskDelegate的
URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
会被调用一次,当下载成功时,error参数为nil。
二. NSURLSession前台下载开始--------->下载取消(cancel)--------->重新恢复下载--------->下载结束。
1.在下载过程中,NSURLSessionDownloadDelegate的
URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didWriteData:(int64_t)bytesWritten totalBytesWritten:(int64_t)totalBytesWritten totalBytesExpectedToWrite (int64_t)totalBytesExpectedToWrite
接口会多次调用,在此接口中可进行下载进度的记录或更新UI的下载进度条。
2.当下载通过cancelByProducingResumeData:接口取消时,会调用NSURLSessionTaskDelegate的
URLSession:(NSURLSession *)session task:(NSURLSessionTask*)task didCompleteWithError:(NSError *)error
一次,此处error不为nil,且可以通过
[error.userInfo objectForKey:NSURLSessionDownloadTaskResumeData]
(键值NSURLSessionDownloadTaskResumeData)取出用于之后恢复本下载任务的resumeData值进行保存以备后用。
3.若要恢复上述下载任务,则可以通过下述方式进恢复(此处用到2中保存的resumeData值),注意,ios10中系统的
downloadTaskWithResumeData接口会有问题,因此需自己进行实现,具体示例查看最终给出的参考文章链接中的demo
- (void)continueDownload {
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;
}
}
4.当调用3中的方法进行断点恢复下载时,会调用NSURLSessionDownloadDelegate接口的方法。
- (void)URLSession:(NSURLSession *)session
downloadTask:(NSURLSessionDownloadTask *)downloadTask
didResumeAtOffset:(int64_t)fileOffset
expectedTotalBytes:(int64_t)expectedTotalBytes
5.当下载顺利结束时,NSURLSessionDownloadDelegate的
URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
didFinishDownloadingToURL:(NSURL *)location
接口会被调用一次,此接口中的location参数记录下载的tmp文件的存储路径,在此接口中需将下载的tmp文件重新保存至最终的保存地址中。
6.若下载成功或失败时,NSURLSessionTaskDelegate的
URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
会被调用一次,当下载成功时,error参数为nil。
三. NSURLSession前台下载开始--------->下载失败--------->重新恢复下载--------->下载结束。
1.在下载过程中,NSURLSessionDownloadDelegate的
接口会多次调用,在此接口中可进行下载进度的记录或更新UI的下载进度条。
URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didWriteData:(int64_t)bytesWritten totalBytesWritten:(int64_t)totalBytesWritten totalBytesExpectedToWrite (int64_t)totalBytesExpectedToWrite
2.当下载失败时,NSURLSessionTaskDelegate的
/*
* 该方法下载成功和失败都会回调,只是失败的是error是有值的,
* 在下载失败时,error的userinfo属性可以通过NSURLSessionDownloadTaskResumeData
* 这个key来取到resumeData(和上面的resumeData是一样的),再通过resumeData恢复下载
*/
- (void)URLSession:(NSURLSession *)session
task:(NSURLSessionTask *)task
didCompleteWithError:(NSError *)error {
NSLog(@"------这是NSURLSessionTaskDelegate 的 didCompleteWithError ");
if (error) {
// check if resume data are available
if ([error.userInfo objectForKey:NSURLSessionDownloadTaskResumeData]) {
NSData *resumeData = [error.userInfo objectForKey:NSURLSessionDownloadTaskResumeData];
//通过之前保存的resumeData,获取断点的NSURLSessionTask,调用resume恢复下载
self.resumeData = resumeData;
}
} else {
[self sendLocalNotification];
[self postDownlaodProgressNotification:@"1"];
}
}
会被调用一次,当下载成功时,此时error不为nil,且可以通过其中的NSURLSessionDownloadTaskResumeData值,得到下载任务的resumedata。
3.若要恢复下载,则可以通过上述得到的resumedata恢复下载:
- (void)continueDownload {
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;
}
}
4.此后的调用过程与上相同。
四.NSURLSession前台下载开始--------->应用退出--------->应用重新打开--------->重新恢复下载--------->下载结束。
1.在下载过程中,NSURLSessionDownloadDelegate的
URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didWriteData:(int64_t)bytesWritten totalBytesWritten:(int64_t)totalBytesWritten totalBytesExpectedToWrite (int64_t)totalBytesExpectedToWrite
接口会多次调用,在此接口中可进行下载进度的记录或更新UI的下载进度条。
2. 杀掉应用,在应用被杀掉前,iOS系统保存应用下载sesson的信息。
3. 重新启动应用,执行AppDelegate的以下方法
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
当创建和之前相同identifier的session时,苹果通过identifier找到对应的session数据,并回调NSURLSessionTaskDelegate的代理方法
/*
* 该方法下载成功和失败都会回调,只是失败的是error是有值的,
* 在下载失败时,error的userinfo属性可以通过NSURLSessionDownloadTaskResumeData
* 这个key来取到resumeData(和上面的resumeData是一样的),再通过resumeData恢复下载
*/
- (void)URLSession:(NSURLSession *)session
task:(NSURLSessionTask *)task
didCompleteWithError:(NSError *)error {
NSLog(@"------这是NSURLSessionTaskDelegate 的 didCompleteWithError ");
if (error) {
// check if resume data are available
if ([error.userInfo objectForKey:NSURLSessionDownloadTaskResumeData]) {
NSData *resumeData = [error.userInfo objectForKey:NSURLSessionDownloadTaskResumeData];
//通过之前保存的resumeData,获取断点的NSURLSessionTask,调用resume恢复下载
self.resumeData = resumeData;
}
} else {
[self sendLocalNotification];
[self postDownlaodProgressNotification:@"1"];
}
}
并在其中通过error取到杀死应用之前保存的关于session的resumedata信息,之后可以通过resumedata进行恢复下载。
4.当调用3中的方法进行断点恢复下载时,会调用NSURLSessionDownloadDelegate接口的方法。
- (void)URLSession:(NSURLSession *)session
downloadTask:(NSURLSessionDownloadTask *)downloadTask
didResumeAtOffset:(int64_t)fileOffset
expectedTotalBytes:(int64_t)expectedTotalBytes
5.当下载顺利结束时,NSURLSessionDownloadDelegate的
URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
didFinishDownloadingToURL:(NSURL *)location
接口会被调用一次,此接口中的location参数记录下载的tmp文件的存储路径,在此接口中需将下载的tmp文件重新保存至最终的保存地址中。
6.若下载成功或失败时,NSURLSessionTaskDelegate的
URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
会被调用一次,当下载成功时,error参数为nil。
五.NSURLSession前台下载开始--------->下载失败--------->应用退出--------->应用重新打开--------->重新恢复下载--------->下载结束。
1.在下载过程中,NSURLSessionDownloadDelegate的
URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didWriteData:(int64_t)bytesWritten totalBytesWritten:(int64_t)totalBytesWritten totalBytesExpectedToWrite (int64_t)totalBytesExpectedToWrite
接口会多次调用,在此接口中可进行下载进度的记录或更新UI的下载进度条。
2.当下载失败时,NSURLSessionTaskDelegate的
/*
* 该方法下载成功和失败都会回调,只是失败的是error是有值的,
* 在下载失败时,error的userinfo属性可以通过NSURLSessionDownloadTaskResumeData
* 这个key来取到resumeData(和上面的resumeData是一样的),再通过resumeData恢复下载
*/
- (void)URLSession:(NSURLSession *)session
task:(NSURLSessionTask *)task
didCompleteWithError:(NSError *)error {
NSLog(@"------这是NSURLSessionTaskDelegate 的 didCompleteWithError ");
if (error) {
// check if resume data are available
if ([error.userInfo objectForKey:NSURLSessionDownloadTaskResumeData]) {
NSData *resumeData = [error.userInfo objectForKey:NSURLSessionDownloadTaskResumeData];
//通过之前保存的resumeData,获取断点的NSURLSessionTask,调用resume恢复下载
self.resumeData = resumeData;
}
} else {
[self sendLocalNotification];
[self postDownlaodProgressNotification:@"1"];
}
}
会被调用一次,当下载成功时,此时error不为nil,且可以通过其中的NSURLSessionDownloadTaskResumeData值,得到下载任务的resumedata。
3.若要恢复下载,则可以通过上述得到的resumedata恢复下载:
- (void)continueDownload {
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;
}
}
4. 杀掉应用,在应用被杀掉前,iOS系统保存应用下载sesson的信息。
5. 重新启动应用,执行AppDelegate的以下方法
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
当创建和之前相同identifier的session时,苹果通过identifier找到对应的session数据,并回调NSURLSessionTaskDelegate的代理方法
/*
* 该方法下载成功和失败都会回调,只是失败的是error是有值的,
* 在下载失败时,error的userinfo属性可以通过NSURLSessionDownloadTaskResumeData
* 这个key来取到resumeData(和上面的resumeData是一样的),再通过resumeData恢复下载
*/
- (void)URLSession:(NSURLSession *)session
task:(NSURLSessionTask *)task
didCompleteWithError:(NSError *)error {
NSLog(@"------这是NSURLSessionTaskDelegate 的 didCompleteWithError ");
if (error) {
// check if resume data are available
if ([error.userInfo objectForKey:NSURLSessionDownloadTaskResumeData]) {
NSData *resumeData = [error.userInfo objectForKey:NSURLSessionDownloadTaskResumeData];
//通过之前保存的resumeData,获取断点的NSURLSessionTask,调用resume恢复下载
self.resumeData = resumeData;
}
} else {
[self sendLocalNotification];
[self postDownlaodProgressNotification:@"1"];
}
}
并在其中通过error取到杀死应用之前保存的关于session的resumedata信息,之后可以通过resumedata进行恢复下载。
6.当调用3中的方法进行断点恢复下载时,会调用NSURLSessionDownloadDelegate接口的方法。
- (void)URLSession:(NSURLSession *)session
downloadTask:(NSURLSessionDownloadTask *)downloadTask
didResumeAtOffset:(int64_t)fileOffset
expectedTotalBytes:(int64_t)expectedTotalBytes
7.当下载顺利结束时,NSURLSessionDownloadDelegate的
URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask
didFinishDownloadingToURL:(NSURL *)location
接口会被调用一次,此接口中的location参数记录下载的tmp文件的存储路径,在此接口中需将下载的tmp文件重新保存至最终的保存地址中。
8.若下载成功或失败时,NSURLSessionTaskDelegate的
URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
会被调用一次,当下载成功时,error参数为nil。