NSURLSession前台下载/后台下载/断点下载的代理执行路径(一)

本文详细介绍了iOS中使用NSURLSession进行后台下载的过程,包括如何配置后台下载会话、处理下载的不同状态(如开始、取消、失败及恢复)、以及如何在应用退出后重启下载任务。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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。



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值