OStack程序员社区-中国程序员成长平台

标题: ios - 使用 NSURLSessionDownloadTask 逐个下载文件 [打印本页]

作者: 菜鸟教程小白    时间: 2022-12-12 12:06
标题: ios - 使用 NSURLSessionDownloadTask 逐个下载文件

在我的应用程序中,我从 Web 服务获取 JSON。此 JSON 包含我要下载的几个文件的 url。

我想使用 NSURLSessionDownloadTask 一个一个地下载每个文件(等待第一个下载完成,直到第二个开始,依此类推)。我还想跟踪写入的总字节数,以便更新 UI。

提前非常感谢!



Best Answer-推荐答案


众所周知,NSURLSessionDownloadTask 与 NSOperationQueues 的配合不太好,这与它们的对应物 NSURLConnection 不同(它可以封装在 NSOperation 中)。

一种选择是将所有 url 添加到一个数组中,然后在任务的 completionHandler 中,只需将下一个项目排队即可。

因此,您可以在循环中创建任务,在每个任务完成处理程序中调用 progressBlock,将任务存储在数组中,然后在每个任务完成处理程序中排队下一个任务:

- (void)addRequestsWithURLsNSArray *)urls
              progressBlockvoid (^)(NSUInteger numberOfFinishedOperations, NSUInteger totalNumberOfOperations, NSURLSessionDownloadTask *task,NSURL *location, NSURLResponse *response, NSError *error))progressBlock {
    __block NSUInteger numberOfFinishedOperations = 0;
    NSUInteger totalNumberOfOperations = [urls count];
    for (NSString *url in urls) {
        NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:url]];
        __block NSURLSessionDownloadTask *task = [self.session downloadTaskWithRequest:request
                                                                completionHandler:^(NSURL *location, NSURLResponse *response, NSError *error) {
                                                                   //downloadFileSomewhere

                                                                    ++numberOfFinishedOperations;
                                                                        if (progressBlock) {
                                                                            progressBlock(numberOfFinishedOperations, totalNumberOfOperations,task,destination != nil ? [NSURL fileURLWithPath:destination] : nil,response,error);

                                                                        }
                                                                    //queueNext
                                                                    [self processCompletedTask:task];
                                                                }];
        //stores an array of NSURLSessionTasks
        [self.tasksWaitingToBeQueued addObject:task];
    }

}

- (void)processCompletedTaskNSURLSessionTask *)completedTask {
    //clean up and queue next one
    [self.tasksWaitingToBeQueued removeObject:completedTask];        
    nextTask = [self.tasksWaitingToBeQueued firstObject];
    if (nextTask) {
        [nextTask resume];
    }
}

注意

在此示例中,我将进度显示为完成的任务数而不是字节数,这是推荐的方法(也更简单)。要使用字节指示进度,您需要事先知道要下载的字节总数(因为您想显示进度条),还需要实现 NSURLSession 委托(delegate)并监控每个任务的进度,捕获下载的字节并更新您的 block .如果您的服务器没有告诉您总字节数,那么您可能需要对每个资源进行 HEAD 请求并汇总大小。就个人而言,这种解决方案很复杂,因为可以通过将进度指示为下载的文件数量来解决。

要实现这一点,可能看起来像这样:

- (void)URLSessionNSURLSession *)session 
      downloadTaskNSURLSessionDownloadTask *)downloadTask
      didWriteDataint64_t)bytesWritten
 totalBytesWrittenint64_t)totalBytesWritten
 totalBytesExpectedToWriteint64_t)totalBytesExpectedToWrite {
    self.totalBytesWritten += totalBytesWritten;
    NSUInteger totalProgressSoFar = self.totalBytesWritten;
    NSUInteger totalExpectedBytes = self.totalExpectedBytes;
    //you would need to capture some progress block locally - beware of retain cycles
    self.progressBlock(totalProgressSoFar/totalExpectedBytes)
}

当你完成后,你应该将 progressBlock 设置为 nil 到 prevent any retain cycles .

关于ios - 使用 NSURLSessionDownloadTask 逐个下载文件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26297367/






欢迎光临 OStack程序员社区-中国程序员成长平台 (https://ostack.cn/) Powered by Discuz! X3.4