我的iOS应用程序以下列方式工作
NSURLSessionDataTask
POST请求,它返回一个字符串,该字符串是要下载的文件的URL。(此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"];
数据代表下载(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]];
}
}
}
简而言之,在数据任务完成后不会调用downloadtask。据我所知,NSURLSession
配置为backgroundsession后,操作系统承担了下载的责任,所以理想情况下它应该可以工作。我错过了什么吗?
问题源于iOS在应用程序处于后台时不会让您启动新任务的事实。我相信这样做是为了防止应用程序不断启动新任务并使应用程序无限期保持活动状态。
后台任务通常适用于在任务正在进行时优雅地处理从前台切换到后台的应用程序。这就是你看到第一个任务完成的原因,而不是第二个任务。
一种解决方法是在您启动初始数据任务时从操作系统请求后台执行时间:
UIBackgroundTaskIdentifier backgroundTaskID = [[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:^{
}];
这应该至少为你的应用程序提供几分钟的后台时间来启动第二项任务。在第二个任务上调用-endBackgroundTask:
后,请务必调用-resume
。
不幸的是,这并不能保证您的所有文件都会下载。如果您发现自己没有时间,那么最好使用Silent Push Notifications唤醒您的应用并定期下载。