【问题标题】:NSURLSessionDataTask not called after NSURLSessionDownloadTask in the same NSURLSession in iOS backgroundNSURLSessionDataTask 在 iOS 后台的同一个 NSURLSession 中的 NSURLSessionDownloadTask 之后没有被调用
【发布时间】:2017-09-02 18:10:02
【问题描述】:

我的 iOS 应用程序以下列方式工作

  1. 我有一个音频文件名数组,我检查文件是否存在。如果它存在,它将开始播放。完成后,我开始播放下一个音频。

  2. 如果文件不存在,我会发出 NSURLSessionDataTask POST 请求,该请求返回一个字符串,该字符串是要下载的文件的 URL。(此 URL 对一个有效分钟)。

  3. 收到 URL 后,我会发出 NSURLSessionDownloadTask 请求并下载文件、保存文件并播放音频。

当应用程序处于前台时,整个过程运行良好。 当所有音频都存在并且应用程序在后台运行时,也可以正常工作。

当音频文件不存在并且应用程序在后台运行时会出现问题

我的代码: BackgroundFetchManager.h

@interface BackgroundFetchManager : NSObject <NSURLSessionDownloadDelegate,NSURLSessionDataDelegate>

@property bool isDownloading;
@property NSURLSessionDownloadTask *download;
@property NSURLSessionDataTask *dataDownload;
@property (nonatomic, strong)NSURLSession *backgroundSession;
@property NSMutableArray *downloadQueue;
@property FileAccessManager *fileAccessManager;

@end

BackgroundFetchManager.m

初始化函数

-(id) init
{
    self = [super init];
    NSURLSessionConfiguration *config = [NSURLSessionConfiguration backgroundSessionConfigurationWithIdentifier:@"testConfiguration"];
    config.sessionSendsLaunchEvents = YES;
    config.discretionary = YES;
    self.backgroundSession = [NSURLSession sessionWithConfiguration:config delegate:self delegateQueue:[NSOperationQueue mainQueue]];
    download = [NSURLSessionDownloadTask new];
    dataDownload = [NSURLSessionDataTask new];
    return self;
} 

数据任务

dataDownload = [self.backgroundSession dataTaskWithRequest:request];
[dataDownload setTaskDescription:@"Data task"];

dataDownload 的代表 (NSURLSessionDataTask)

- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask
didReceiveResponse:(NSURLResponse *)response
 completionHandler:(void (^)(NSURLSessionResponseDisposition disposition))completionHandler
{

    completionHandler(NSURLSessionResponseAllow);
}

- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask
    didReceiveData:(NSData *)data
{
    NSString * str = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
    NSString *url = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
    NSLog(@"This is the response received %@",url);

//download is the NSURLSessionDownloadTask
    download = [self.backgroundSession downloadTaskWithURL:[NSURL URLWithString:url]];
    [download resume];

}

下载代表 (NSURLSessionDownloadTask)

这些是应用在后台时不会调用的那些

- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didFinishDownloadingToURL:(NSURL *)location
{
    AppDelegate *a = (AppDelegate *)[[UIApplication sharedApplication] delegate];
    NSURL *destinationURL = [fileAccessManager saveFile:a.downloadQueue[0] fromLocation:location]; // save file code
}


- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
{
    // audio playing stuff

    if (![task.taskDescription isEqualToString:@"Data task"]) {
        NSLog(@"This is did complete with error %@", task );
        AppDelegate *a = (AppDelegate *)[[UIApplication sharedApplication] delegate];
        if ([a.downloadQueue[0] isEqualToString:a.audioToBePlayed]) {
            NSDictionary* userInfo = @{@"file": a.downloadQueue[0]};
            [[NSNotificationCenter defaultCenter] postNotificationName:@"playThisStuff" object:nil userInfo:userInfo];


        }
//starts next download
        [a pop];
        isDownloading = false;
        if ([a.downloadQueue count] > 0) {
            isDownloading = true;
            [self downloadFile:a.downloadQueue[0]];
        }
    }        
}

所以简而言之,在数据任务完成后不会调用下载任务。 据我所知,NSURLSession 当配置为后台会话时,操作系统负责下载,所以理想情况下它应该可以工作。我错过了什么吗?

【问题讨论】:

    标签: ios objective-c multitasking nsurlsessiondownloadtask background-thread


    【解决方案1】:

    问题源于 iOS 不允许您在应用处于后台时启动新任务。我相信这样做是为了防止应用程序不断启动新任务并让应用程序无限期地保持活动状态。

    后台任务通常适合在任务进行时优雅地处理您的应用从前台转换到后台。这就是您看到第一个任务完成但第二个没有完成的原因。

    一种解决方法是在您启动初始数据任务时从操作系统请求后台执行时间:

    UIBackgroundTaskIdentifier backgroundTaskID = [[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:^{
    
    }];
    

    这至少应该让您的应用有几分钟的后台时间来启动第二个任务。在第二个任务上调用-resume 后,请务必调用-endBackgroundTask:

    很遗憾,这不能保证您的所有文件都会下载。如果您发现后台时间已用完,最好使用Silent Push Notifications 唤醒您的应用并定期下载。

    【讨论】:

    • 我到办公室后会检查一下。
    猜你喜欢
    • 1970-01-01
    • 2017-01-31
    • 1970-01-01
    • 1970-01-01
    • 2014-01-03
    • 1970-01-01
    • 2015-09-12
    • 1970-01-01
    • 2021-05-19
    相关资源
    最近更新 更多